source: spip-zone/_plugins_/formidable/trunk/traiter/email.php @ 112372

Last change on this file since 112372 was 112372, checked in by maieul@…, 23 months ago

commentaire php/yaml explicitant r110181

File size: 18.2 KB
Line 
1<?php
2
3// Sécurité
4if (!defined('_ECRIRE_INC_VERSION')) {
5        return;
6}
7include_spip('inc/utils');
8include_spip('inc/formidable_fichiers');
9
10function traiter_email_dist($args, $retours) {
11        if (!isset($retours['fichiers'])) {
12                $retours['fichiers'] = array();
13                $ajouter_fichier = true;
14        } else {
15                $ajouter_fichier = false;
16        }
17        $timestamp = time();
18        $retours['timestamp'] = $timestamp;
19        $formulaire = $args['formulaire'];
20        $options = $args['options'];
21        $saisies = unserialize($formulaire['saisies']);
22        $traitements = unserialize($formulaire['traitements']);
23        $champs = saisies_lister_champs($saisies);
24        $destinataires = array();
25        $taille_fichiers = 0; //taille des fichiers en email
26        $fichiers_facteur = array(); // tableau qui stockera les fichiers à envoyer avec facteur
27        if (isset($options['exclure_champs_email']) && $options['exclure_champs_email']) {
28                $champs_a_exclure = explode(",", $options['exclure_champs_email']);
29                $champs = array_diff($champs, $champs_a_exclure);
30                foreach ($champs_a_exclure as $champ_a_exclure) {
31                        $champ_a_exclure = trim($champ_a_exclure);
32                        $champ_a_exclure = str_replace('@','',$champ_a_exclure);
33                        $saisies = saisies_supprimer($saisies,$champ_a_exclure);
34                }
35        }
36
37        $nom_site_spip = supprimer_tags(typo(lire_meta('nom_site')));
38        $email_webmaster = supprimer_tags(typo(lire_meta('email_webmaster')));
39
40        // On récupère les destinataires
41        if ($options['champ_destinataires']) {
42                $destinataires = _request($options['champ_destinataires']);
43                if (!is_array($destinataires)) {
44                        if (intval($destinataires)) {
45                                $destinataires = array($destinataires);
46                        } else {
47                                $destinataires = array();
48                        }
49                }
50                if (count($destinataires)) {
51                        // On récupère les mails des destinataires
52                        $destinataires = array_map('intval', $destinataires);
53                        $destinataires = sql_allfetsel(
54                                'email',
55                                'spip_auteurs',
56                                sql_in('id_auteur', $destinataires)
57                        );
58                        $destinataires = array_map('reset', $destinataires);
59                }
60        }
61
62        // Conservé pour raison historique, mais mauvaise pratique.
63        // Cf fichier .yam
64        if ($options['champ_courriel_destinataire_form']) {
65                $courriel_champ_form = _request($options['champ_courriel_destinataire_form']);
66                $destinataires[] = $courriel_champ_form;
67        }
68
69
70        // On ajoute les destinataires en plus
71        if ($options['destinataires_plus']) {
72                $destinataires_plus = explode(',', $options['destinataires_plus']);
73                $destinataires_plus = array_map('trim', $destinataires_plus);
74                $destinataires = array_merge($destinataires, $destinataires_plus);
75                $destinataires = array_unique($destinataires);
76        }
77
78        // On ajoute les destinataires en fonction des choix de saisie dans le formulaire
79        // @selection_1@/choix1 : mail@domain.tld
80        // @selection_1@/choix2 : autre@domain.tld, lapin@domain.tld
81        if (!empty($options['destinataires_selon_champ'])) {
82                if ($destinataires_selon_champ = formidable_traiter_email_destinataire_selon_champ($options['destinataires_selon_champ'])) {
83                        $destinataires = array_merge($destinataires, $destinataires_selon_champ);
84                        $destinataires = array_unique($destinataires);
85                }
86        }
87
88        // On récupère le courriel de l'envoyeur s'il existe
89        if ($options['champ_courriel']) {
90                $courriel_envoyeur = _request($options['champ_courriel']);
91        }
92        if (!isset($courriel_envoyeur)) {
93                $courriel_envoyeur = '';
94        }
95
96        // Récuperer les valeurs saisies
97        list($valeurs,$valeurs_libellees) = formidable_tableau_valeurs_saisies($saisies);
98
99        // Traitement à part pour les saisies de types fichiers :
100        // 1. Calculer la taille totale des fichiers
101        // 2. Ajouter au facteur
102        $saisies_fichiers = saisies_lister_avec_type($saisies, 'fichiers');
103        foreach ($champs as $champ) {
104                if (array_key_exists($champ, $saisies_fichiers)) {
105                        $valeurs[$champ] = traiter_email_fichiers($saisies_fichiers[$champ], $champ, $formulaire['id_formulaire'], $retours, $timestamp);
106                        if ($ajouter_fichier) {
107                                $retours['fichiers'][$champ] = $valeurs[$champ];
108                        }
109                        $taille_fichiers += formidable_calculer_taille_fichiers_saisie($valeurs[$champ]);
110                        $fichiers_facteur = array_merge(
111                                $fichiers_facteur,
112                                vue_fichier_to_tableau_facteur($valeurs[$champ])
113                        );
114                }
115        }
116        // Si on a bien des destinataires, on peut continuer
117        if ($destinataires or ($courriel_envoyeur and $options['activer_accuse'])) {
118                include_spip('inc/filtres');
119                include_spip('inc/texte');
120
121                // On récupère le nom de l'envoyeur
122                if ($options['champ_nom']) {
123                        $nom_envoyeur = formidable_raccourcis_arobases_2_valeurs_champs(
124                                $options['champ_nom'],
125                                $saisies
126                        );
127                }
128                if (!isset($nom_envoyeur) or !$nom_envoyeur) {
129                        $nom_envoyeur = $nom_site_spip;
130                }
131
132                // On récupère le sujet s'il existe sinon on le construit
133                if ($options['champ_sujet']) {
134                        $sujet = formidable_raccourcis_arobases_2_valeurs_champs(
135                                $options['champ_sujet'],
136                                $saisies,
137                                $options['champ_sujet_valeurs_brutes']
138                        );
139                }
140                if (!isset($sujet) or !$sujet) {
141                        $sujet = _T('formidable:traiter_email_sujet', array('nom'=>$nom_envoyeur));
142                }
143                $sujet = filtrer_entites($sujet);
144
145                // Mais quel va donc être le fond ?
146                if (find_in_path('notifications/formulaire_'.$formulaire['identifiant'].'_email.html')) {
147                        $notification = 'notifications/formulaire_'.$formulaire['identifiant'].'_email';
148                } else {
149                        $notification = 'notifications/formulaire_email';
150                }
151                // Est-ce qu'on est assez léger pour joindre les pj
152                $joindre_pj = false;
153                if ($taille_fichiers < 1024 * 1024 * _FORMIDABLE_TAILLE_MAX_FICHIERS_EMAIL
154                        and
155                        $traitements['email']['pj'] == 'on'
156                ) {
157                        $joindre_pj = true;
158                        foreach (array_keys($saisies_fichiers) as $nom) {
159                                $saisies = saisies_supprimer($saisies,$nom);
160                        }
161                }
162                // On génère le mail avec le fond
163                $html = recuperer_fond(
164                        $notification,
165                        array(
166                                'id_formulaire' => $args['id_formulaire'],
167                                'id_formulaires_reponse' => isset($retours['id_formulaires_reponse']) ? $retours['id_formulaires_reponse'] : '',
168                                'titre' => _T_ou_typo($formulaire['titre']),
169                                'traitements' => $traitements,
170                                'saisies' => $saisies,
171                                'valeurs' => $valeurs,
172                                'masquer_liens' => $options['masquer_liens'],
173                                'ip' => $options['activer_ip']?$GLOBALS['ip']:'',
174                                'courriel_envoyeur' => $courriel_envoyeur,
175                                'nom_envoyeur' => filtrer_entites($nom_envoyeur)
176                        )
177                );
178
179                // On génère le texte brut
180                include_spip('facteur_fonctions');
181                $texte = facteur_mail_html2text($html);
182
183                // On utilise la forme avancée de Facteur
184                $corps = array(
185                        'html' => $html,
186                        'texte' => $texte,
187                        'nom_envoyeur' => filtrer_entites($nom_envoyeur),
188                );
189                // Joindre les pj si léger
190                if ($joindre_pj) {
191                        $corps['pieces_jointes'] = $fichiers_facteur;
192                }
193
194                // Si l'utilisateur n'a pas indiqué autrement, on met le courriel de l'envoyeur dans
195                // Reply-To et on laisse le from par defaut de Facteur car sinon ca bloque sur les
196                // SMTP un peu restrictifs.
197                $courriel_from = '';
198                if ($courriel_envoyeur && $options['activer_vrai_envoyeur']) {
199                        $courriel_from = $courriel_envoyeur;
200                } elseif ($courriel_envoyeur) {
201                        $corps['repondre_a'] = $courriel_envoyeur;
202                }
203
204                // On envoie enfin le message
205                $envoyer_mail = charger_fonction('envoyer_mail', 'inc');
206
207                // On envoie aux destinataires
208                if ($destinataires) {
209                        $ok = $envoyer_mail($destinataires, $sujet, $corps, $courriel_from, 'X-Originating-IP: '.$GLOBALS['ip']);
210                }
211
212                // Si c'est bon, on envoie l'accusé de réception
213                if ($ok and $courriel_envoyeur and $options['activer_accuse']) {
214                        // On récupère le sujet s'il existe sinon on le construit
215                        if ($options['sujet_accuse']) {
216                                $sujet_accuse = formidable_raccourcis_arobases_2_valeurs_champs(
217                                        $options['sujet_accuse'],
218                                        $saisies
219                                );
220                        }
221                        if (!isset($sujet_accuse) or !$sujet_accuse) {
222                                $sujet_accuse = _T('formidable:traiter_email_sujet_accuse');
223                        }
224                        $sujet_accuse = filtrer_entites($sujet_accuse);
225
226                        // Mais quel va donc être le fond ?
227                        if (find_in_path('notifications/formulaire_'.$formulaire['identifiant'].'_accuse.html')) {
228                                $accuse = 'notifications/formulaire_'.$formulaire['identifiant'].'_accuse';
229                        } else {
230                                $accuse = 'notifications/formulaire_accuse';
231                        }
232
233                        // On génère l'accusé de réception
234                        if (_FORMIDABLE_LIENS_FICHIERS_ACCUSE_RECEPTION == false) {
235                                $valeurs = vues_saisies_supprimer_action_recuperer_fichier_par_email($saisies, $valeurs);
236                        }
237                        $html_accuse = recuperer_fond(
238                                $accuse,
239                                array(
240                                        'id_formulaire' => $formulaire['id_formulaire'],
241                                        'id_formulaires_reponse' => isset($retours['id_formulaires_reponse']) ? $retours['id_formulaires_reponse'] : '',
242                                        'titre' => _T_ou_typo($formulaire['titre']),
243                                        'message_retour' => formidable_raccourcis_arobases_2_valeurs_champs(
244                                                $formulaire['message_retour'],
245                                                $saisies),
246                                        'traitements' => $traitements,
247                                        'saisies' => $saisies,
248                                        'valeurs' => $valeurs
249                                )
250                        );
251
252                        // On génère le texte brut
253                        $texte = facteur_mail_html2text($html_accuse);
254
255                        // Si un nom d'expéditeur est précisé pour l'AR, on l'utilise,
256                        // sinon on utilise le nomde l'envoyeur du courriel principal
257                        $nom_envoyeur_accuse = trim($options['nom_envoyeur_accuse']);
258                        if (!$nom_envoyeur_accuse) {
259                                $nom_envoyeur_accuse = $nom_site_spip;
260                        }
261
262                        $corps = array(
263                                'html' => $html_accuse,
264                                'texte' => $texte,
265                                'nom_envoyeur' => filtrer_entites($nom_envoyeur_accuse),
266                        );
267
268                        //A fortiori, si un courriel d'expéditeur est précisé pour l'AR, on l'utilise
269                        if ($options['courriel_envoyeur_accuse']) {
270                                $courriel_envoyeur_accuse = $options['courriel_envoyeur_accuse'];
271                        } else {
272                                $courriel_envoyeur_accuse = $email_webmaster;
273                        }
274
275                        //Et on teste si on doit mettre cela en from ou en reply-to
276                        if ($options['activer_vrai_envoyeur'] and $courriel_envoyeur_accuse) {
277                                $courriel_from_accuse = $courriel_envoyeur_accuse;
278                        } elseif ($courriel_envoyeur_accuse) {
279                                $corps['repondre_a'] = $courriel_envoyeur_accuse;
280                                $courriel_from_accuse = '';
281                        }
282
283                        // Joindre les pj si léger et nécessaire
284                        if ($joindre_pj and  _FORMIDABLE_LIENS_FICHIERS_ACCUSE_RECEPTION == false) {
285                                $corps['pieces_jointes'] = $fichiers_facteur;
286                        }
287
288                        $ok = $envoyer_mail($courriel_envoyeur, $sujet_accuse, $corps, $courriel_from_accuse, 'X-Originating-IP: '.$GLOBALS['ip']);
289                }
290
291                if ($ok) {
292                        if (isset($retours['message_ok'])) {
293                                $retours['message_ok'] .= "\n"._T('formidable:traiter_email_message_ok');
294                        } else {
295                                $retours['message_ok'] = _T('formidable:traiter_email_message_ok');
296                        }
297                } else {
298                        if (isset($retours['message_erreur'])) {
299                                $retours['message_erreur'] .= "\n"._T('formidable:traiter_email_message_erreur');
300                        } else {
301                                $retours['message_erreur'] = _T('formidable:traiter_email_message_erreur');
302                        }
303                }
304        }
305
306        // noter qu'on a deja fait le boulot, pour ne pas risquer double appel
307        $retours['traitements']['email'] = true;
308        return $retours;
309}
310
311
312/**
313 * Retourne la liste des destinataires sélectionnés en fonction
314 * de l'option 'destinataires_selon_champ' du traitement email.
315 *
316 * @param string $description
317 *     Description saisie dans l'option du traitement du formulaire,
318 *     qui respecte le schéma prévu, c'est à dire : 1 description par ligne,
319 *     tel que `@champ@/valeur : mail@domain.tld, mail@domain.tld, ...`
320 *     {@example : `@selection_2@/choix_1 : toto@domain.tld`}
321 * @return array
322 *     Liste des destinataires, s'il y en a.
323 **/
324function formidable_traiter_email_destinataire_selon_champ($description) {
325        $destinataires = array();
326
327        // 1 test à rechercher par ligne
328        $descriptions = explode("\n", trim($description));
329        $descriptions = array_map('trim', $descriptions);
330        $descriptions = array_filter($descriptions);
331
332        // pour chaque test, s'il est valide, ajouter les courriels indiqués
333        foreach ($descriptions as $test) {
334                // Un # est un commentaire
335                if ($test[0] == '#') {
336                        continue;
337                }
338                // Le premier caractère est toujours un @
339                if ($test[0] != '@') {
340                        continue;
341                }
342
343
344                list($champ, $reste) = explode('/', $test, 2);
345                $champ = substr(trim($champ), 1, -1); // enlever les @
346
347                if ($reste) {
348                        list($valeur, $mails) = explode(':', $reste, 2);
349                        $valeur = trim($valeur);
350                        $mails = explode(',', $mails);
351                        $mails = array_map('trim', $mails);
352                        $mails = array_filter($mails);
353                        if ($mails) {
354                                // obtenir la valeur du champ saisi dans le formulaire.
355                                // cela peut être un tableau.
356                                $champ = _request($champ);
357                                if (!is_null($champ)) {
358                                        $ok = is_array($champ) ? in_array($valeur, $champ) : ($champ == $valeur);
359
360                                        if ($ok) {
361                                                $destinataires = array_merge($destinataires, $mails);
362                                                $destinataires = array_unique($destinataires);
363                                        }
364                                }
365                        }
366                }
367        }
368
369        return $destinataires;
370}
371
372/**
373 * Gère une saisie de type fichiers dans le traitement par email.
374 * C'est à dire:
375 *      - S'il y a eu un enregistement avant, ne déplace pas le fichier
376 *      - S'il n'y a pas eu d'enregistrement avant, déplace le fichier
377 *              dans un dossier nommé en fonction du timestamp du traitement
378 *      - Renvoie un tableau décrivant les fichiers, avec une url d'action sécurisée valable seulement
379 *              _FORMIDABLE_EXPIRATION_FICHIERS_EMAIL (sauf si cette constantes est définie à 0)
380 * @param array $saisie la description de la saisie
381 * @param string $nom le nom de la saisie
382 * @param int|string $id_formulaire le formulaire concerné
383 * @param array $retours ce qu'a envoyé le précédent traitement
384 * @param int $timestamp un timestamp correspondant au début du processus de création du courriel
385 * @return array un tableau décrivant la saisie
386 **/
387function traiter_email_fichiers($saisie, $nom, $id_formulaire, $retours, $timestamp) {
388        //Initialisation
389        $id_formulaire = strval($id_formulaire);//précaution
390        $vue = array();
391
392        if (isset($retours['id_formulaires_reponse']) and $id_formulaires_reponse = $retours['id_formulaires_reponse']) { // cas simple: les réponses ont été enregistrées
393                if (isset($retours['fichiers'][$nom])) { // petite précaution
394                        $options = array(
395                                'id_formulaire' => $id_formulaire,
396                                'id_formulaires_reponse' => $retours['id_formulaires_reponse']
397                        );
398                        $vue = ajouter_action_recuperer_fichier_par_email($retours['fichiers'][$nom], $nom, $options);
399                }
400        } else { // si les réponses n'ont pas été enregistrées
401                $vue = formidable_deplacer_fichiers_produire_vue_saisie($saisie, array('id_formulaire' => $id_formulaire, 'timestamp' => $timestamp));
402                        $options = array(
403                                'id_formulaire' => $id_formulaire,
404                                'timestamp' => $timestamp
405                        );
406                        $vue = ajouter_action_recuperer_fichier_par_email($vue, $nom, $options);
407        }
408
409        return $vue;
410}
411
412
413/**
414 * Pour une saisie de type 'fichiers'
415 * insère dans la description du résultat de cette saisie
416 * l'url de l'action pour récuperer la saisie par email
417 * Ajoute également une vignette correspondant à l'extension
418 * @param array $saisie_a_modifier
419 * @param string $nom_saisie
420 * @param array $options options qui décrit l'endroit où est stocké le fichier
421 * @return array $saisie_a_modifier
422 **/
423function ajouter_action_recuperer_fichier_par_email($saisie_a_modifier, $nom_saisie, $options) {
424        $vignette_par_defaut = charger_fonction('vignette', 'inc/');
425
426        if (_FORMIDABLE_EXPIRATION_FICHIERS_EMAIL > 0) {
427                $delai = secondes_en_jour(_FORMIDABLE_EXPIRATION_FICHIERS_EMAIL);
428        }
429        foreach ($saisie_a_modifier as $i => $valeur) {
430                $url = formidable_generer_url_action_recuperer_fichier_email($nom_saisie, $valeur['nom'], $options);
431                $saisie_a_modifier[$i]['url'] = $url;
432                if (_FORMIDABLE_EXPIRATION_FICHIERS_EMAIL > 0) {
433                        $saisie_a_modifier[$i]['fichier'] = $valeur['nom'];
434                        $saisie_a_modifier[$i]['nom'] = '['._T('formidable:lien_expire', array('delai' => $delai)).'] '.$valeur['nom'];
435                } else {
436                        $saisie_a_modifier[$i]['fichier'] = $valeur['nom'];
437                        $saisie_a_modifier[$i]['nom'] = $valeur['nom'];
438                }
439                if (isset($valeur['extension'])) {
440                        $saisie_a_modifier[$i]['vignette'] = $vignette_par_defaut($valeur['extension'], false);
441                }
442        }
443        return $saisie_a_modifier;
444}
445/**
446 * Supprime dans une vue de saisie 'fichiers'
447 * l'url de récupération par email
448 * et l'information sur le délai d'expiration
449 * @param array $vue
450 * @return array $vue
451**/
452function supprimer_action_recuperer_fichier_par_email($vue) {
453        foreach ($vue as $f => &$desc) {
454                if (isset($desc['url'])) {
455                        unset($desc['url']);
456                }
457                $desc['nom'] = $desc['fichier'];
458        }
459        return $vue;
460}
461
462/**
463 * Dans l'ensemble de vues des saisies
464 * recherche les saisies 'fichiers'
465 * et supprime pour chacune d'entre elle les actions de récupération de fichier
466 * @param array $saisies
467 * @param array $vues
468 * @return array $vues
469**/
470function vues_saisies_supprimer_action_recuperer_fichier_par_email($saisies, $vues) {
471        foreach ($saisies as $saisie => $description) {
472                if ($description['saisie'] == 'fichiers') { // si de type fichiers
473                        $nom_saisie = $description['options']['nom'];
474                        $vues[$nom_saisie] = supprimer_action_recuperer_fichier_par_email($vues[$nom_saisie]);
475                }
476        }
477        return $vues;
478}
479
480/**
481 * Calcule la taille totale des fichiers
482 * d'après une saisie de type fichiers
483 * @param array $saisie
484 * @return int $taille (en octets)
485**/
486function formidable_calculer_taille_fichiers_saisie($saisie) {
487        $taille = 0;
488        foreach ($saisie as $k => $info) {
489                $taille += $info['taille'];
490        }
491        return $taille;
492}
493
494/**
495 * Converti une description d'une vue fichiers en description passable à facteur
496 * @param array $vue
497 * @return array $tableau_facteur
498**/
499function vue_fichier_to_tableau_facteur($vue) {
500        $tableau_facteur = array();
501        foreach ($vue as $fichier) {
502                $arg = unserialize(parametre_url($fichier['url'],'arg'));
503                $tableau_facteur[] = array(
504                        'chemin' => formidable_generer_chemin_fichier($arg),
505                        'nom' => $fichier['fichier'],
506                        'encodage' => 'base64',
507                        'mime' => $fichier['mime']);
508        }
509        return $tableau_facteur;
510}
511
512/**
513 * Retourne des secondes sous une jolie forme, du type xx jours, yy heures, zz minutes, aa secondes
514 * @param int $seconde
515 * @return str
516**/
517function secondes_en_jour($secondes) {
518        //On ne peut pas utiliser date_create, car en PHP 5.2, et SPIP 3.0 est à partir de PHP 5.1…
519        $jours = floor($secondes/(24*3600));
520        $heures = floor(($secondes-$jours*24*3600)/3600);
521        $minutes = floor(($secondes-$jours*24*3600-$heures*3600)/60);
522        $secondes = $secondes-$jours*24*3600-$heures*3600-$minutes*60;
523        $param = array(
524                'j' => $jours,
525                'h' => $heures,
526                'm' => $minutes,
527                's' => $secondes
528        );
529        if ($jours > 0) {
530                return _T('formidable:jours_heures_minutes_secondes', $param);
531        } elseif ($heures > 0) {
532                return _T('formidable:heures_minutes_secondes', $param);
533        } elseif ($minutes > 0) {
534                return _T('formidable:minutes_secondes', $param);
535        } else {
536                return _T('formidable:secondes', $param);
537        }
538}
Note: See TracBrowser for help on using the repository browser.