Skip to content
Snippets Groups Projects
CMailFile.class.php 29.2 KiB
Newer Older
Rodolphe Quiedeville's avatar
Rodolphe Quiedeville committed
<?php
/* Copyright (C) 2000-2005 Rodolphe Quiedeville <rodolphe@quiedeville.org>
 * Copyright (C) 2003      Jean-Louis Bergamo   <jlb@j1b.org>
 * Copyright (C) 2004-2009 Laurent Destailleur  <eldy@users.sourceforge.net>
 * Copyright (C) 2005-2009 Regis Houssin        <regis@dolibarr.fr>
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
 * or see http://www.gnu.org/
 *
 * Lots of code inspired from Dan Potter's CMailFile class
 */

/**
Laurent Destailleur's avatar
Laurent Destailleur committed
 *      \file       htdocs/lib/CMailFile.class.php
 *      \brief      File of class to send emails (with attachments or not)
 *		\version    $Id: CMailFile.class.php,v 1.146 2011/07/20 11:07:12 eldy Exp $
Laurent Destailleur's avatar
Laurent Destailleur committed
 *      \author     Dan Potter.
 *      \author	    Eric Seigne
 *      \author	    Laurent Destailleur.
 */
Benoit Mortier's avatar
 
Benoit Mortier committed

Laurent Destailleur's avatar
Laurent Destailleur committed
 *      \class      CMailFile
 *      \brief      Class to send emails (with attachments or not)
Laurent Destailleur's avatar
Laurent Destailleur committed
 *      \remarks    Usage: $mailfile = new CMailFile($subject,$sendto,$replyto,$message,$filepath,$mimetype,$filename,$cc,$ccc,$deliveryreceipt,$msgishtml,$errors_to);
 *      \remarks           $mailfile->sendfile();
 */
Benoit Mortier's avatar
 
Benoit Mortier committed
class CMailFile
Laurent Destailleur's avatar
Laurent Destailleur committed
	var $subject;
	var $addr_from;
	var $errors_to;
	var $addr_to;
	var $addr_cc;
	var $addr_bcc;

	var $mixed_boundary;
	var $related_boundary;
	var $alternative_boundary;
Laurent Destailleur's avatar
Laurent Destailleur committed
	var $deliveryreceipt;

	var $eol;
	var $atleastonefile=0;
	var $error='';

	var $smtps;			// Contains SMTPs object (if this method is used)
	//CSS
	var $css;
	//! Defined css style for body background
	var $styleCSS;
	//! Defined bacckground directly in body tag
	var $bodyCSS;
	var $html;
	var $image_boundary;
	var $atleastoneimage=0;
	var $html_images=array();
	var $images_encoded=array();
	var $image_types = array('gif'  => 'image/gif',
                           'jpg'  => 'image/jpeg',
                           'jpeg' => 'image/jpeg',
                           'jpe'  => 'image/jpeg',
                           'bmp'  => 'image/bmp',
                           'png'  => 'image/png',
                           'tif'  => 'image/tiff',
                           'tiff' => 'image/tiff');
Laurent Destailleur's avatar
Laurent Destailleur committed

	/**
Laurent Destailleur's avatar
Laurent Destailleur committed
	 *	CMailFile
Laurent Destailleur's avatar
Laurent Destailleur committed
	 *	@param 	subject             Topic/Subject of mail
	 *	@param 	to                  Recipients emails (RFC 2822: "Nom prenom <email>[, ...]" ou "email[, ...]" ou "<email>[, ...]")
	 *	@param 	from                Sender email      (RFC 2822: "Nom prenom <email>[, ...]" ou "email[, ...]" ou "<email>[, ...]")
	 *	@param 	msg                 Message
	 *	@param 	filename_list       List of files to attach (full path of filename on file system)
	 *	@param 	mimetype_list       List of MIME type of attached files
	 *	@param 	mimefilename_list   List of attached file name in message
	 *	@param 	addr_cc             Email cc
	 *	@param 	addr_bcc            Email bcc
	 *	@param 	deliveryreceipt		Ask a delivery receipt
	 *	@param 	msgishtml       	1=String IS already html, 0=String IS NOT html, -1=Unknown need autodetection
	 *	@param 	errors_to      		Email errors
	 *	@param	css			        Css option
Laurent Destailleur's avatar
Laurent Destailleur committed
	 */
	function CMailFile($subject,$to,$from,$msg,
	$filename_list=array(),$mimetype_list=array(),$mimefilename_list=array(),
	$addr_cc="",$addr_bcc="",$deliveryreceipt=0,$msgishtml=0,$errors_to='',$css='')
Laurent Destailleur's avatar
Laurent Destailleur committed
	{
		global $conf;
		// We define end of line (RFC 822bis section 2.3)
		$this->eol="\r\n";
		//if (preg_match('/^win/i',PHP_OS)) $this->eol="\r\n";
		//if (preg_match('/^mac/i',PHP_OS)) $this->eol="\r";
		// On defini mixed_boundary
		$this->mixed_boundary = md5(uniqid("dolibarr1"));

		// On defini related_boundary
		$this->related_boundary = md5(uniqid("dolibarr2"));

		// On defini alternative_boundary
		$this->alternative_boundary = md5(uniqid("dolibarr3"));
		// If ending method not defined
		if (empty($conf->global->MAIN_MAIL_SENDMODE)) $conf->global->MAIN_MAIL_SENDMODE='mail';
		dol_syslog("CMailFile::CMailfile: MAIN_MAIL_SENDMODE=".$conf->global->MAIN_MAIL_SENDMODE." charset=".$conf->file->character_set_client." from=$from, to=$to, addr_cc=$addr_cc, addr_bcc=$addr_bcc, errors_to=$errors_to", LOG_DEBUG);
		dol_syslog("CMailFile::CMailfile: subject=$subject, deliveryreceipt=$deliveryreceipt, msgishtml=$msgishtml", LOG_DEBUG);
		// Detect if message is HTML (use fast method)
		}
		else
		{
			$this->msgishtml = $msgishtml;
		}
		// Detect images
		if ($this->msgishtml)
		{
			$this->html = $msg;
			$findimg = $this->findHtmlImages($conf->fckeditor->dir_output);
			// Define if there is at least one file
			if ($findimg)
			{
				foreach ($this->html_images as $i => $val)
				{
					if ($this->html_images[$i])
					{
						$this->atleastoneimage=1;
						dol_syslog("CMailFile::CMailfile: html_images[$i]['name']=".$this->html_images[$i]['name'], LOG_DEBUG);
					}
				}
			}
		}

		// Define if there is at least one file
		foreach ($filename_list as $i => $val)
Laurent Destailleur's avatar
Laurent Destailleur committed
		{
			if ($filename_list[$i])
			{
				$this->atleastonefile=1;
				dol_syslog("CMailFile::CMailfile: filename_list[$i]=".$filename_list[$i].", mimetype_list[$i]=".$mimetype_list[$i]." mimefilename_list[$i]=".$mimefilename_list[$i], LOG_DEBUG);
Benoit Mortier's avatar
 
Benoit Mortier committed

		// Add autocopy to
		if (! empty($conf->global->MAIN_MAIL_AUTOCOPY_TO)) $addr_bcc.=($addr_bcc?', ':'').$conf->global->MAIN_MAIL_AUTOCOPY_TO;

		// Action according to choosed sending method
		if ($conf->global->MAIN_MAIL_SENDMODE == 'mail')
Laurent Destailleur's avatar
Laurent Destailleur committed
		{
			// Use mail php function (default PHP method)
			// ------------------------------------------

			$smtp_headers = "";
			$mime_headers = "";
			$text_body = "";
			$files_encoded = "";
			$this->subject = $subject;
			$this->addr_from = $from;
			$this->errors_to = $errors_to;
			$this->addr_to = $to;
			$this->addr_cc = $addr_cc;
			$this->addr_bcc = $addr_bcc;
			$this->deliveryreceipt = $deliveryreceipt;
			$smtp_headers = $this->write_smtpheaders();

			$mime_headers = $this->write_mimeheaders($filename_list, $mimefilename_list);

			if (! empty($this->html))
			{
				if (!empty($css))
				{
					$this->css = $css;
					$this->buildCSS();
				$msg = $this->html;
			}
			if ($this->atleastoneimage)
			{
				$images_encoded = $this->write_images($this->images_encoded);
				// always end related and end alternative after inline images
				$images_encoded.= "--" . $this->related_boundary . "--" . $this->eol;
				$images_encoded.= $this->eol . "--" . $this->alternative_boundary . "--" . $this->eol;
				$images_encoded.= $this->eol;
			if ($this->atleastonefile)
			{
				$files_encoded = $this->write_files($filename_list,$mimetype_list,$mimefilename_list);
Laurent Destailleur's avatar
Laurent Destailleur committed

			// We now define $this->headers et $this->message
			$this->headers = $smtp_headers . $mime_headers;
			$this->message = $text_body . $images_encoded . $files_encoded;
			$this->message.= "--" . $this->mixed_boundary . "--" . $this->eol;
Laurent Destailleur's avatar
Laurent Destailleur committed

			// On nettoie le header pour qu'il ne se termine pas par un retour chariot.
			// Ceci evite aussi les lignes vides en fin qui peuvent etre interpretees
			// comme des injections mail par les serveurs de messagerie.
			$this->headers = preg_replace("/([\r\n]+)$/i","",$this->headers);
		}
		else if ($conf->global->MAIN_MAIL_SENDMODE == 'smtps')
Laurent Destailleur's avatar
Laurent Destailleur committed
		{
			// Use SMTPS library
			// ------------------------------------------

			require_once(DOL_DOCUMENT_ROOT."/includes/smtps/SMTPs.php");
			$smtps = new SMTPs();
			$smtps->setCharSet($conf->file->character_set_client);

			$smtps->setSubject($this->encodetorfc2822($subject));
Regis Houssin's avatar
Regis Houssin committed
			$smtps->setTO($this->getValidAddress($to,0,1));
			$smtps->setFrom($this->getValidAddress($from,0,1));
				if (!empty($css))
				{
					$this->css = $css;
					$this->styleCSS = $this->buildCSS();
				}
				$msg = $this->html;
				$msg = $this->checkIfHTML($msg);
			}

			if ($this->msgishtml) $smtps->setBodyContent($msg,'html');
			else $smtps->setBodyContent($msg,'plain');

			if ($this->atleastoneimage)
			{
				foreach ($this->images_encoded as $img)
				{
					$smtps->setImageInline($img['image_encoded'],$img['name'],$img['content_type'],$img['cid']);
			if ($this->atleastonefile)
			{
				foreach ($filename_list as $i => $val)
				{
					$content=file_get_contents($filename_list[$i]);
					$smtps->setAttachment($content,$mimefilename_list[$i],$mimetype_list[$i]);
				}
			}

			$smtps->setCC($addr_cc);
			$smtps->setBCC($addr_bcc);
			$smtps->setErrorsTo($errors_to);
			$smtps->setDeliveryReceipt($deliveryreceipt);

			$this->smtps=$smtps;
Laurent Destailleur's avatar
Laurent Destailleur committed
		}
		else
		{
			// Send mail method not correctly defined
			// --------------------------------------

			return 'Bad value for MAIN_MAIL_SENDMODE constant';
Laurent Destailleur's avatar
Laurent Destailleur committed
		}
Laurent Destailleur's avatar
Laurent Destailleur committed
	}

Laurent Destailleur's avatar
Laurent Destailleur committed
	/**
Regis Houssin's avatar
Regis Houssin committed
	 * Send mail that was prepared by constructor
Regis Houssin's avatar
Regis Houssin committed
	 * @return    boolean     True if mail sent, false otherwise
Laurent Destailleur's avatar
Laurent Destailleur committed
	 */
	function sendfile()
	{
		global $conf;
Laurent Destailleur's avatar
Laurent Destailleur committed

		$errorlevel=error_reporting();
		error_reporting($errorlevel ^ E_WARNING);   // Desactive warnings
		$res=false;
Laurent Destailleur's avatar
Laurent Destailleur committed

		if (empty($conf->global->MAIN_DISABLE_ALL_MAILS))
			// Action according to choosed sending method
			if ($conf->global->MAIN_MAIL_SENDMODE == 'mail')
				// Use mail php function (default PHP method)
				// ------------------------------------------
				dol_syslog("CMailFile::sendfile addr_to=".$this->addr_to.", subject=".$this->subject, LOG_DEBUG);
				dol_syslog("CMailFile::sendfile header=\n".$this->headers, LOG_DEBUG);
				//dol_syslog("CMailFile::sendfile message=\n".$message);
				// If Windows, sendmail_from must be defined
				if (isset($_SERVER["WINDIR"]))
				{
					if (empty($this->addr_from)) $this->addr_from = 'robot@mydomain.com';
					@ini_set('sendmail_from',$this->getValidAddress($this->addr_from,2));
				}

				// Forcage parametres
				if (! empty($conf->global->MAIN_MAIL_SMTP_SERVER)) ini_set('SMTP',$conf->global->MAIN_MAIL_SMTP_SERVER);
				if (! empty($conf->global->MAIN_MAIL_SMTP_PORT))   ini_set('smtp_port',$conf->global->MAIN_MAIL_SMTP_PORT);

				$dest=$this->getValidAddress($this->addr_to,2);
				if (! $dest)
					$this->error="Failed to send mail with php mail to HOST=".ini_get('SMTP').", PORT=".ini_get('smtp_port')."<br>Recipient address '$dest' invalid";
					dol_syslog("CMailFile::sendfile: mail end error=".$this->error, LOG_ERROR);
					dol_syslog("CMailFile::sendfile: mail start HOST=".ini_get('SMTP').", PORT=".ini_get('smtp_port'), LOG_DEBUG);
					$bounce = '';	// By default
					if ($conf->global->MAIN_MAIL_ALLOW_SENDMAIL_F)
					{
						// le return-path dans les header ne fonctionne pas avec tous les MTA
						// Le passage par -f est donc possible si la constante MAIN_MAIL_ALLOW_SENDMAIL_F est definie.
						// La variable definie pose des pb avec certains sendmail securisee (option -f refusee car dangereuse)
						$bounce = (! empty($conf->global->MAIN_MAIL_ERRORS_TO) ? '-f' . $conf->global->MAIN_MAIL_ERRORS_TO : ($this->addr_from != '' ? '-f' . $this->addr_from : '') );
					$this->message=stripslashes($this->message);

					if (! empty($conf->global->MAIN_MAIL_DEBUG)) $this->dump_mail();

					if (! empty($bounce)) $res = mail($dest,$this->encodetorfc2822($this->subject),$this->message,$this->headers, $bounce);
					else $res = mail($dest,$this->encodetorfc2822($this->subject),$this->message,$this->headers);
						$this->error="Failed to send mail with php mail to HOST=".ini_get('SMTP').", PORT=".ini_get('smtp_port')."<br>Check your server logs and your firewalls setup";
						dol_syslog("CMailFile::sendfile: mail end error=".$this->error, LOG_ERROR);
						dol_syslog("CMailFile::sendfile: mail end success", LOG_DEBUG);
				}

				if (isset($_SERVER["WINDIR"]))
				{
					@ini_restore('sendmail_from');
				}

				// Forcage parametres
				if (! empty($conf->global->MAIN_MAIL_SMTP_SERVER))	ini_restore('SMTP');
				if (! empty($conf->global->MAIN_MAIL_SMTP_PORT)) 	ini_restore('smtp_port');
			}
			else if ($conf->global->MAIN_MAIL_SENDMODE == 'smtps')
			{

				// Use SMTPS library
				// ------------------------------------------
				$this->smtps->setTransportType(0);	// Only this method is coded in SMTPs library

				// Forcage parametres
				if (empty($conf->global->MAIN_MAIL_SMTP_SERVER)) $conf->global->MAIN_MAIL_SMTP_SERVER=ini_get('SMTP');
				if (empty($conf->global->MAIN_MAIL_SMTP_PORT))   $conf->global->MAIN_MAIL_SMTP_PORT=ini_get('smtp_port');

				// If we use SSL/TLS
				$server=$conf->global->MAIN_MAIL_SMTP_SERVER;
				if (! empty($conf->global->MAIN_MAIL_EMAIL_TLS) && function_exists('openssl_open')) $server='ssl://'.$server;

				$this->smtps->setHost($server);
				$this->smtps->setPort($conf->global->MAIN_MAIL_SMTP_PORT); // 25, 465...;
				if (! empty($conf->global->MAIN_MAIL_SMTPS_ID)) $this->smtps->setID($conf->global->MAIN_MAIL_SMTPS_ID);
				if (! empty($conf->global->MAIN_MAIL_SMTPS_PW)) $this->smtps->setPW($conf->global->MAIN_MAIL_SMTPS_PW);
				//$smtps->_msgReplyTo  = 'reply@web.com';
				$res=true;
				$from=$this->smtps->getFrom('org');
				if (! $from)
				{
					$this->error="Failed to send mail with smtps lib to HOST=".$server.", PORT=".$conf->global->MAIN_MAIL_SMTP_PORT."<br>Sender address '$from' invalid";
					dol_syslog("CMailFile::sendfile: mail end error=".$this->error, LOG_ERR);
					$res=false;
				}
				$dest=$this->smtps->getTo();
				if (! $dest)
					$this->error="Failed to send mail with smtps lib to HOST=".$server.", PORT=".$conf->global->MAIN_MAIL_SMTP_PORT."<br>Recipient address '$dest' invalid";
					dol_syslog("CMailFile::sendfile: mail end error=".$this->error, LOG_ERR);
				if ($res)
					if (! empty($conf->global->MAIN_MAIL_DEBUG)) $this->smtps->setDebug(true);
					$result=$this->smtps->sendMsg();
					if (! empty($conf->global->MAIN_MAIL_DEBUG)) $this->dump_mail();
					$result=$this->smtps->getErrors();
					if (empty($this->error) && empty($result)) $res=true;
					else
					{
						if (empty($this->error)) $this->error=$result;
						dol_syslog("CMailFile::sendfile: mail end error=".$this->error, LOG_ERR);
						$res=false;
					}

				// Send mail method not correctly defined
				// --------------------------------------

				return 'Bad value for MAIN_MAIL_SENDMODE constant';
			$this->error='No mail sent. Feature is disabled by option MAIN_DISABLE_ALL_MAILS';
			dol_syslog("CMailFile::sendfile: ".$this->error, LOG_WARNING);
		error_reporting($errorlevel);              // Reactive niveau erreur origine
		return $res;
	}
	// Encode subject according to RFC 2822 - http://en.wikipedia.org/wiki/MIME#Encoded-Word
	function encodetorfc2822($stringtoencode)
	{
		global $conf;
		return '=?'.$conf->file->character_set_client.'?B?'.base64_encode($stringtoencode).'?=';
	}
	 * Read a file on disk and return encoded content for emails (mode = 'mail')
Regis Houssin's avatar
Regis Houssin committed
	 * @param      sourcefile
	 * @return     <0 if KO, encoded string if OK
	 */
	function _encode_file($sourcefile)
	{
		$newsourcefile=utf8_check($sourcefile)?utf8_decode($sourcefile):$sourcefile;	// is_readable and file_get_contents need ISO filename

		if (is_readable($newsourcefile))
			$contents = file_get_contents($newsourcefile);	// Need PHP 4.3
			$encoded = chunk_split(base64_encode($contents), 68, $this->eol);
			return $encoded;
		}
		else
		{
			$this->error="Error: Can't read file '$sourcefile'";
			dol_syslog("CMailFile::encode_file: ".$this->error, LOG_ERR);
Laurent Destailleur's avatar
Laurent Destailleur committed
	/**
	 *  Write content of a SMTP request into a dump file (mode = all)
	 *  Used for debugging.
Laurent Destailleur's avatar
Laurent Destailleur committed
	 */
	function dump_mail()
	{
		global $conf,$dolibarr_main_data_root;

Laurent Destailleur's avatar
Laurent Destailleur committed
		if (@is_writeable($dolibarr_main_data_root))	// Avoid fatal error on fopen with open_basedir
		{
			$fp = fopen($dolibarr_main_data_root."/dolibarr_mail.log","w");

			if ($conf->global->MAIN_MAIL_SENDMODE == 'mail')
			{
				fputs($fp, $this->headers);
				fputs($fp, $this->eol);			// This eol is added by the mail function, so we add it in log
				fputs($fp, $this->message);
			}
			elseif ($conf->global->MAIN_MAIL_SENDMODE == 'smtps')
			{
				fputs($fp, $this->smtps->log);
			}

Laurent Destailleur's avatar
Laurent Destailleur committed
			fclose($fp);
			if (! empty($conf->global->MAIN_UMASK))
			@chmod($outputfile, octdec($conf->global->MAIN_UMASK));
		}
	}


    /**
     * Correct an uncomplete html string
     *
     * @param       $msg
     * @return
     */
    function checkIfHTML($msg)
    {
        if (!preg_match('/^[\s\t]*<html/i',$msg))
        {
            $out = "<html><head><title></title>";
            if (!empty($this->styleCSS)) $out.= $this->styleCSS;
            $out.= "</head><body";
            if (!empty($this->bodyCSS)) $out.= $this->bodyCSS;
            $out.= ">";
            $out.= $msg;
            $out.= "</body></html>";
        }
        else
        {
            $out = $msg;
        }

        return $out;
    }

    /**
     * Build a css style (mode = all)
     *
     * @return css
     */
    function buildCSS()
    {
        if (! empty($this->css))
        {
            // Style CSS
            $this->styleCSS = '<style type="text/css">';
            $this->styleCSS.= 'body {';

            if ($this->css['bgcolor'])
            {
                $this->styleCSS.= '  background-color: '.$this->css['bgcolor'].';';
                $this->bodyCSS.= ' BGCOLOR="'.$this->css['bgcolor'].'"';
            }
            if ($this->css['bgimage'])
            {
                // TODO recuperer cid
                $this->styleCSS.= ' background-image: url("cid:'.$this->css['bgimage_cid'].'");';
            }
            $this->styleCSS.= '}';
            $this->styleCSS.= '</style>';
        }
    }


Laurent Destailleur's avatar
Laurent Destailleur committed
	/**
Regis Houssin's avatar
Regis Houssin committed
	 * @return	smtp headers
Laurent Destailleur's avatar
Laurent Destailleur committed
	 */
	function write_smtpheaders()
	{
Laurent Destailleur's avatar
Laurent Destailleur committed
		//$out .= "X-Sender: ".getValidAddress($this->addr_from,2).$this->eol;
		$out .= "From: ".$this->getValidAddress($this->addr_from,0,1).$this->eol;
		$out .= "Return-Path: ".$this->getValidAddress($this->addr_from,0,1).$this->eol;
		if (isset($this->reply_to)  && $this->reply_to)  $out .= "Reply-To: ".$this->getValidAddress($this->reply_to,2).$this->eol;
		if (isset($this->errors_to) && $this->errors_to) $out .= "Errors-To: ".$this->getValidAddress($this->errors_to,2).$this->eol;
		if (isset($this->addr_cc)   && $this->addr_cc)   $out .= "Cc: ".$this->getValidAddress($this->addr_cc,2).$this->eol;
		if (isset($this->addr_bcc)  && $this->addr_bcc)  $out .= "Bcc: ".$this->getValidAddress($this->addr_bcc,2).$this->eol;
Laurent Destailleur's avatar
Laurent Destailleur committed

		// Accuse reception
		if (isset($this->deliveryreceipt) && $this->deliveryreceipt == 1) $out .= "Disposition-Notification-To: ".$this->getValidAddress($this->addr_from,2).$this->eol;
Laurent Destailleur's avatar
Laurent Destailleur committed

		//$out .= "X-Priority: 3".$this->eol;
		$out.= "X-Mailer: Dolibarr version " . DOL_VERSION ." (using php mail)".$this->eol;
		$out.= "MIME-Version: 1.0".$this->eol;
		$out.= "Content-Type: multipart/mixed; boundary=\"".$this->mixed_boundary."\"".$this->eol;
		$out.= "Content-Transfer-Encoding: 8bit".$this->eol;
		dol_syslog("CMailFile::write_smtpheaders smtp_header=\n".$out);
Laurent Destailleur's avatar
Laurent Destailleur committed
		return $out;
	}
Laurent Destailleur's avatar
Laurent Destailleur committed
	/**
	 * Create header MIME (mode = 'mail')
Regis Houssin's avatar
Regis Houssin committed
	 * @param 		filename_list
	 * @param 		mimefilename_list
	 * @return		mime headers
Laurent Destailleur's avatar
Laurent Destailleur committed
	 */
	function write_mimeheaders($filename_list, $mimefilename_list)
	{
Laurent Destailleur's avatar
Laurent Destailleur committed
		$out = "";
Laurent Destailleur's avatar
Laurent Destailleur committed
		{
Philippe Grand's avatar
Philippe Grand committed
			$filename_list_size=count($filename_list);
			for($i=0;$i < $filename_list_size;$i++)
Laurent Destailleur's avatar
Laurent Destailleur committed
			{
					if ($mimefilename_list[$i]) $filename_list[$i] = $mimefilename_list[$i];
					$out.= "X-attachments: $filename_list[$i]".$this->eol;
Laurent Destailleur's avatar
Laurent Destailleur committed
			}
		}
		dol_syslog("CMailFile::write_mimeheaders mime_header=\n".$out, LOG_DEBUG);
Laurent Destailleur's avatar
Laurent Destailleur committed
		return $out;
	}

	/**
	 * Return email content (mode = 'mail')
Regis Houssin's avatar
Regis Houssin committed
	 * @param 		msgtext
Laurent Destailleur's avatar
Laurent Destailleur committed
	 */
Laurent Destailleur's avatar
Laurent Destailleur committed
	{
		global $conf;

		$out='';
		if ($this->atleastoneimage)
Laurent Destailleur's avatar
Laurent Destailleur committed
		{
			$out.= "--" . $this->mixed_boundary . $this->eol;
			$out.= "Content-Type: multipart/alternative; boundary=\"".$this->alternative_boundary."\"".$this->eol;
			$out.= $this->eol;
			$out.= "--" . $this->alternative_boundary . $this->eol;
			$out.= "--" . $this->mixed_boundary . $this->eol;
Laurent Destailleur's avatar
Laurent Destailleur committed
		}
Laurent Destailleur's avatar
Laurent Destailleur committed
		if ($this->msgishtml)
		{
			// Check if html header already in message
			$strContent = $this->checkIfHTML($msgtext);
Laurent Destailleur's avatar
Laurent Destailleur committed
		}
		else
		{
Laurent Destailleur's avatar
Laurent Destailleur committed
		}
		// Make RFC821 Compliant, replace bare linefeeds
		$strContent = preg_replace("/(?<!\r)\n/si", "\r\n", $strContent );
        //$strContent = rtrim(chunk_split($strContent));    // Function chunck_split seems bugged
        $strContent = rtrim(wordwrap($strContent));
		if ($this->msgishtml)
		{
			if ($this->atleastoneimage)
			{
				$out.= "Content-Type: text/plain; charset=".$conf->file->character_set_client.$this->eol;
				$out.= $this->eol.strip_tags($strContent).$this->eol; // Add plain text message
				$out.= "--" . $this->alternative_boundary . $this->eol;
				$out.= "Content-Type: multipart/related; boundary=\"".$this->related_boundary."\"".$this->eol;
				$out.= $this->eol;
				$out.= "--" . $this->related_boundary . $this->eol;
			}
			$out.= "Content-Type: text/html; charset=".$conf->file->character_set_client.$this->eol;
			$out.= $this->eol.$strContent.$this->eol;
		}
		else
		{
			$out.= "Content-Type: text/plain; charset=".$conf->file->character_set_client.$this->eol;
			$out.= $this->eol.$strContent.$this->eol;
		$out.= $this->eol;
Laurent Destailleur's avatar
Laurent Destailleur committed
		return $out;
	}
Laurent Destailleur's avatar
Laurent Destailleur committed
	/**
	 * Attach file to email (mode = 'mail')
Regis Houssin's avatar
Regis Houssin committed
	 * @param 		filename_list		Tableau
	 * @param 		mimetype_list		Tableau
	 * @param 		mimefilename_list	Tableau
	 * @return		out					Chaine fichiers encodes
Laurent Destailleur's avatar
Laurent Destailleur committed
	 */
	function write_files($filename_list,$mimetype_list,$mimefilename_list)
	{
		$out = '';

Philippe Grand's avatar
Philippe Grand committed
		$filename_list_size=count($filename_list);
		for($i=0;$i < $filename_list_size;$i++)
Laurent Destailleur's avatar
Laurent Destailleur committed
		{
			if ($filename_list[$i])
			{
				dol_syslog("CMailFile::write_files: i=$i");
Laurent Destailleur's avatar
Laurent Destailleur committed
				$encoded = $this->_encode_file($filename_list[$i]);
				if ($encoded >= 0)
				{
					if ($mimefilename_list[$i]) $filename_list[$i] = $mimefilename_list[$i];
					if (! $mimetype_list[$i]) { $mimetype_list[$i] = "application/octet-stream"; }

					$out.= "--" . $this->mixed_boundary . $this->eol;
                    $out.= "Content-Disposition: attachment; filename=\"".$filename_list[$i]."\"".$this->eol;
Laurent Destailleur's avatar
Laurent Destailleur committed
					$out.= "Content-Type: " . $mimetype_list[$i] . "; name=\"".$filename_list[$i]."\"".$this->eol;
					$out.= "Content-Transfer-Encoding: base64".$this->eol;
					$out.= "Content-Description: File Attachment".$this->eol;
Laurent Destailleur's avatar
Laurent Destailleur committed
					$out.= $this->eol;
					$out.= $encoded;
					$out.= $this->eol;
Laurent Destailleur's avatar
Laurent Destailleur committed
				}
				else
				{
					return $encoded;
				}
Laurent Destailleur's avatar
Laurent Destailleur committed
		}
	 * Attach an image to email (mode = 'mail')
Regis Houssin's avatar
Regis Houssin committed
	 * @param 		images_list		Tableau
	 * @return		out				Chaine images encodees
	 */
	function write_images($images_list)
	{
		$out = '';

		if ($images_list)
		{
			foreach ($images_list as $img)
			{
				dol_syslog("CMailFile::write_images: i=$i");

				$out.= "--" . $this->related_boundary . $this->eol; // always related for an inline image
				$out.= "Content-Type: " . $img["content_type"] . "; name=\"".$img["name"]."\"".$this->eol;
				$out.= "Content-Transfer-Encoding: base64".$this->eol;
				$out.= "Content-Disposition: inline; filename=\"".$img["name"]."\"".$this->eol;
				$out.= "Content-ID: <".$img["cid"].">".$this->eol;
				$out.= $this->eol;
				$out.= $img["image_encoded"];
				$out.= $this->eol;
			}
		}
Laurent Destailleur's avatar
Laurent Destailleur committed
		return $out;
	}


	/**
	 * Try to create a socket connection
Laurent Destailleur's avatar
Laurent Destailleur committed
	 * @param 		$host		Add ssl:// for SSL/TLS.
	 * @param 		$port		Example: 25, 465
	 * @return 		Socket id if ok, 0 if KO
	 */
	function check_server_port($host,$port)
	{
		$_retVal=0;
		$timeout=5;	// Timeout in seconds

		if (function_exists('fsockopen'))
		{
Laurent Destailleur's avatar
Laurent Destailleur committed
			dol_syslog("Try socket connection to host=".$host." port=".$port);
			//See if we can connect to the SMTP server
			if ( $socket = @fsockopen($host,       // Host to test, IP or domain. Add ssl:// for SSL/TLS.
Laurent Destailleur's avatar
Laurent Destailleur committed
			$port,       // which Port number to use
			$errno,      // actual system level error
			$errstr,     // and any text that goes with the error
			$timeout) )  // timeout for reading/writing data over the socket
Laurent Destailleur's avatar
Laurent Destailleur committed
			{
				// Windows still does not have support for this timeout function
				if (function_exists('stream_set_timeout')) stream_set_timeout($socket, $timeout, 0);
Laurent Destailleur's avatar
Laurent Destailleur committed

				dol_syslog("Now we wait for answer 220");

				// Check response from Server
				if ( $_retVal = $this->server_parse($socket, "220") ) $_retVal = $socket;
			}
			else
			{
				$this->error = 'Error '.$errno.' - '.$errstr;
			}
		}
		return $_retVal;
	}

	 * This function has been modified as provided by SirSir to allow multiline responses when
	 * using SMTP Extensions.
	 * @param      socket
	 * @param      response
	 * @return     boolean
	 */
	function server_parse($socket, $response)
	{
Laurent Destailleur's avatar
Laurent Destailleur committed
		$_retVal = true;	// Indicates if Object was created or not
		$server_response = '';

		while ( substr($server_response,3,1) != ' ' )
		{
			if( !( $server_response = fgets($socket, 256) ) )
			{
				$this->error="Couldn't get mail server response codes";
				$_retVal = false;
			}
		}

		if( !( substr($server_response, 0, 3) == $response ) )
		{
			$this->error="Ran into problems sending Mail.\r\nResponse: $server_response";
			$_retVal = false;
		}

		return $_retVal;
Laurent Destailleur's avatar
Laurent Destailleur committed
	}
	 * Seearch images into html message and init array this->images_encoded if found
	 * @param 		images_dir		Location of physical images files
Regis Houssin's avatar
Regis Houssin committed
	 * @return		int         	>0 if OK, <0 if KO
	{
		// Build the list of image extensions
		$extensions = array_keys($this->image_types);

		preg_match_all('/(?:"|\')([^"\']+\.('.implode('|', $extensions).'))(?:"|\')/Ui', $this->html, $matches);

		if ($matches)
		{
			$i=0;
			foreach ($matches[1] as $full)
			{
				if (preg_match('/file=([A-Za-z0-9_\-\/]+[\.]?[A-Za-z0-9]+)?$/i',$full,$regs))
					if (file_exists($images_dir.'/'.$img))
					{
						// Image path in src
						$src = preg_quote($full,'/');
						// Image full path
						$this->html_images[$i]["fullpath"] = $images_dir.'/'.$img;
						// Image name
						$this->html_images[$i]["name"] = $img;
						// Content type
						$ext = preg_replace('/^.*\.(\w{3,4})$/e', 'strtolower("$1")', $img);
						$this->html_images[$i]["content_type"] = $this->image_types[$ext];
						// cid
						$this->html_images[$i]["cid"] = md5(uniqid(time()));
						$this->html = preg_replace("/src=\"$src\"|src='$src'/i", "src=\"cid:".$this->html_images[$i]["cid"]."\"", $this->html);
					}
					$i++;
					// If duplicate images are embedded, they may show up as attachments, so remove them.
					if (!in_array($fullpath,$inline))
						// Read image file
						if ($image = file_get_contents($fullpath))
						{
							// On garde que le nom de l'image
							preg_match('/([A-Za-z0-9_-]+[\.]?[A-Za-z0-9]+)?$/i',$img["name"],$regs);
							$this->images_encoded[$i]['name'] = $imgName;
							$this->images_encoded[$i]['content_type'] = $img["content_type"];
							$this->images_encoded[$i]['cid'] = $img["cid"];
							// Encodage de l'image
							$this->images_encoded[$i]["image_encoded"] = chunk_split(base64_encode($image), 68, $this->eol);
							$inline[] = $fullpath;
						}
Regis Houssin's avatar
Regis Houssin committed
	 * Return an address for SMTP protocol
	 * @param       adresses		Example: 'John Doe <john@doe.com>' or 'john@doe.com'
Regis Houssin's avatar
Regis Houssin committed
	 * @param		format			0=Auto, 1=emails with <>, 2=emails without <>
	 * @param		encode			1=Encode name to RFC2822
	 * @return	    string			If format 1: '<john@doe.com>' or 'John Doe <john@doe.com>'
	 *								If format 2: 'john@doe.com'
	 */
	function getValidAddress($adresses,$format,$encode='')
	{
		global $conf;
Laurent Destailleur's avatar
Laurent Destailleur committed

		$arrayaddress=explode(',',$adresses);
Laurent Destailleur's avatar
Laurent Destailleur committed

		// Boucle sur chaque composant de l'adresse
		foreach($arrayaddress as $val)
			if (preg_match('/^(.*)<(.*)>$/i',trim($val),$regs))
				$name  = trim($regs[1]);
				$email = trim($regs[2]);
				$newemail='';
				if ($format == 2)
				{
					$newemail=$email;
				}
				if ($format == 1)
				{
Regis Houssin's avatar
Regis Houssin committed
					$newemail='<'.$email.'>';
				}
				if ($format == 0)
				{
					if ($conf->global->MAIN_MAIL_NO_FULL_EMAIL) $newemail='<'.$email.'>';
					elseif (! $name) $newemail='<'.$email.'>';
					else $newemail=($encode?$this->encodetorfc2822($name):$name).' <'.$email.'>';
				}
Laurent Destailleur's avatar
Laurent Destailleur committed

Rodolphe Quiedeville's avatar
Rodolphe Quiedeville committed
}