Source for file Deliver.class.php

Documentation is available at Deliver.class.php

  1. <?php
  2.  
  3. /**
  4.  * Deliver.class.php
  5.  *
  6.  * This contains all the functions needed to send messages through
  7.  * a delivery backend.
  8.  *
  9.  * @author Marc Groot Koerkamp
  10.  * @copyright &copy; 1999-2006 The SquirrelMail Project Team
  11.  * @license http://opensource.org/licenses/gpl-license.php GNU Public License
  12.  * @version $Id: Deliver.class.php,v 1.65 2006/07/15 12:00:44 tokul Exp $
  13.  * @package squirrelmail
  14.  */
  15.  
  16. /**
  17.  * Deliver Class - called to actually deliver the message
  18.  *
  19.  * This class is called by compose.php and other code that needs
  20.  * to send messages.  All delivery functionality should be centralized
  21.  * in this class.
  22.  *
  23.  * Do not place UI code in this class, as UI code should be placed in templates
  24.  * going forward.
  25.  *
  26.  * @author  Marc Groot Koerkamp
  27.  * @package squirrelmail
  28.  */
  29. class Deliver {
  30.  
  31.     /**
  32.      * function mail - send the message parts to the SMTP stream
  33.      *
  34.      * @param Message  $message  Message class to send
  35.      * @param resource $stream   file handle to the SMTP stream
  36.      *
  37.      * @return integer $raw_length
  38.      */
  39.     function mail($message$stream=false{
  40.         $rfc822_header $message->rfc822_header;
  41.         if (count($message->entities)) {
  42.             $boundary $this->mimeBoundary();
  43.             $rfc822_header->content_type->properties['boundary']='"'.$boundary.'"';
  44.         else {
  45.             $boundary='';
  46.         }
  47.         $raw_length 0;
  48.         $reply_rfc822_header (isset($message->reply_rfc822_header)
  49.                              ? $message->reply_rfc822_header '');
  50.         $header $this->prepareRFC822_Header($rfc822_header$reply_rfc822_header$raw_length);
  51.  
  52.         if ($stream{
  53.             $this->preWriteToStream($header);
  54.             $this->writeToStream($stream$header);
  55.         }
  56.         $this->writeBody($message$stream$raw_length$boundary);
  57.         return $raw_length;
  58.     }
  59.  
  60.     /**
  61.      * function writeBody - generate and write the mime boundaries around each part to the stream
  62.      *
  63.      * Recursively formats and writes the MIME boundaries of the $message
  64.      * to the output stream.
  65.      *
  66.      * @param Message   $message      Message object to transform
  67.      * @param resource  $stream       SMTP output stream
  68.      * @param integer  &$length_raw   raw length of the message (part)
  69.      *                                 as returned by mail fn
  70.      * @param string    $boundary     custom boundary to call, usually for subparts
  71.      *
  72.      * @return void 
  73.      */
  74.     function writeBody($message$stream&$length_raw$boundary=''{
  75.         // calculate boundary in case of multidimensional mime structures
  76.         if ($boundary && $message->entity_id && count($message->entities)) {
  77.             if (strpos($boundary,'_part_')) {
  78.                 $boundary substr($boundary,0,strpos($boundary,'_part_'));
  79.  
  80.             // the next four lines use strrev to reverse any nested boundaries
  81.             // because RFC 2046 (5.1.1) says that if a line starts with the outer
  82.             // boundary string (doesn't matter what the line ends with), that
  83.             // can be considered a match for the outer boundary; thus the nested
  84.             // boundary needs to be unique from the outer one
  85.             //
  86.             else if (strpos($boundary,'_trap_')) {
  87.                 $boundary substr(strrev($boundary),0,strpos(strrev($boundary),'_part_'));
  88.             }
  89.             $boundary_new strrev($boundary '_part_'.$message->entity_id);
  90.         else {
  91.             $boundary_new $boundary;
  92.         }
  93.         if ($boundary && !$message->rfc822_header{
  94.             $s '--'.$boundary."\r\n";
  95.             $s .= $this->prepareMIME_Header($message$boundary_new);
  96.             $length_raw += strlen($s);
  97.             if ($stream{
  98.                 $this->preWriteToStream($s);
  99.                 $this->writeToStream($stream$s);
  100.             }
  101.         }
  102.         $this->writeBodyPart($message$stream$length_raw);
  103.  
  104.         $last false;
  105.         for ($i=0$entCount=count($message->entities);$i<$entCount;$i++{
  106.             $msg $this->writeBody($message->entities[$i]$stream$length_raw$boundary_new);
  107.             if ($i == $entCount-1$last true;
  108.         }
  109.         if ($boundary && $last{
  110.             $s "--".$boundary_new."--\r\n\r\n";
  111.             $length_raw += strlen($s);
  112.             if ($stream{
  113.                 $this->preWriteToStream($s);
  114.                 $this->writeToStream($stream$s);
  115.             }
  116.         }
  117.     }
  118.  
  119.     /**
  120.      * function writeBodyPart - write each individual mimepart
  121.      *
  122.      * Recursively called by WriteBody to write each mime part to the SMTP stream
  123.      *
  124.      * @param Message   $message      Message object to transform
  125.      * @param resource  $stream       SMTP output stream
  126.      * @param integer  &$length       length of the message part
  127.      *                                 as returned by mail fn
  128.      *
  129.      * @return void 
  130.      */
  131.     function writeBodyPart($message$stream&$length{
  132.         if ($message->mime_header{
  133.             $type0 $message->mime_header->type0;
  134.         else {
  135.             $type0 $message->rfc822_header->content_type->type0;
  136.         }
  137.  
  138.         $body_part_trailing $last '';
  139.         switch ($type0)
  140.         {
  141.         case 'text':
  142.         case 'message':
  143.             if ($message->body_part{
  144.                 $body_part $message->body_part;
  145.                 // remove NUL characters
  146.                 $body_part str_replace("\0",'',$body_part);
  147.                 $length += $this->clean_crlf($body_part);
  148.                 if ($stream{
  149.                     $this->preWriteToStream($body_part);
  150.                     $this->writeToStream($stream$body_part);
  151.                 }
  152.                 $last $body_part;
  153.             elseif ($message->att_local_name{
  154.                 $filename $message->att_local_name;
  155.                 $file fopen ($filename'rb');
  156.                 while ($body_part fgets($file4096)) {
  157.                     // remove NUL characters
  158.                     $body_part str_replace("\0",'',$body_part);
  159.                     $length += $this->clean_crlf($body_part);
  160.                     if ($stream{
  161.                         $this->preWriteToStream($body_part);
  162.                         $this->writeToStream($stream$body_part);
  163.                     }
  164.                     $last $body_part;
  165.                 }
  166.                 fclose($file);
  167.             }
  168.             break;
  169.         default:
  170.             if ($message->body_part{
  171.                 $body_part $message->body_part;
  172.                 // remove NUL characters
  173.                 $body_part str_replace("\0",'',$body_part);
  174.                 $length += $this->clean_crlf($body_part);
  175.                 if ($stream{
  176.                     $this->writeToStream($stream$body_part);
  177.                 }
  178.             elseif ($message->att_local_name{
  179.                 $filename $message->att_local_name;
  180.                 $file fopen ($filename'rb');
  181.                 while ($tmp fread($file570)) {
  182.                     $body_part chunk_split(base64_encode($tmp));
  183.                     // Up to 4.3.10 chunk_split always appends a newline,
  184.                     // while in 4.3.11 it doesn't if the string to split
  185.                     // is shorter than the chunk length.
  186.                     ifsubstr($body_part-!= "\n" )
  187.                         $body_part .= "\n";
  188.                     $length += $this->clean_crlf($body_part);
  189.                     if ($stream{
  190.                         $this->writeToStream($stream$body_part);
  191.                     }
  192.                 }
  193.                 fclose($file);
  194.             }
  195.             break;
  196.         }
  197.         $body_part_trailing '';
  198.         if ($last && substr($last,-1!= "\n"{
  199.             $body_part_trailing "\r\n";
  200.         }
  201.         if ($body_part_trailing{
  202.             $length += strlen($body_part_trailing);
  203.             if ($stream{
  204.                 $this->preWriteToStream($body_part_trailing);
  205.                 $this->writeToStream($stream$body_part_trailing);
  206.             }
  207.         }
  208.     }
  209.  
  210.     /**
  211.      * function clean_crlf - change linefeeds and newlines to legal characters
  212.      *
  213.      * The SMTP format only allows CRLF as line terminators.
  214.      * This function replaces illegal teminators with the correct terminator.
  215.      *
  216.      * @param string &$s string to clean linefeeds on
  217.      *
  218.      * @return void 
  219.      */
  220.     function clean_crlf(&$s{
  221.         $s str_replace("\r\n""\n"$s);
  222.         $s str_replace("\r""\n"$s);
  223.         $s str_replace("\n""\r\n"$s);
  224.         return strlen($s);
  225.     }
  226.  
  227.     /**
  228.      * function strip_crlf - strip linefeeds and newlines from a string
  229.      *
  230.      * The SMTP format only allows CRLF as line terminators.
  231.      * This function strips all line terminators from the string.
  232.      *
  233.      * @param string &$s string to clean linefeeds on
  234.      *
  235.      * @return void 
  236.      */
  237.     function strip_crlf(&$s{
  238.         $s str_replace("\r\n "''$s);
  239.         $s str_replace("\r"''$s);
  240.         $s str_replace("\n"''$s);
  241.     }
  242.  
  243.     /**
  244.      * function preWriteToStream - reserved for extended functionality
  245.      *
  246.      * This function is not yet implemented.
  247.      * Reserved for extended functionality.
  248.      *
  249.      * @param string &$s string to operate on
  250.      *
  251.      * @return void 
  252.      */
  253.     function preWriteToStream(&$s{
  254.     }
  255.  
  256.     /**
  257.      * function writeToStream - write data to the SMTP stream
  258.      *
  259.      * @param resource $stream  SMTP output stream
  260.      * @param string   $data    string with data to send to the SMTP stream
  261.      *
  262.      * @return void 
  263.      */
  264.     function writeToStream($stream$data{
  265.         fputs($stream$data);
  266.     }
  267.  
  268.     /**
  269.      * function initStream - reserved for extended functionality
  270.      *
  271.      * This function is not yet implemented.
  272.      * Reserved for extended functionality.
  273.      *
  274.      * @param Message $message  Message object
  275.      * @param string  $host     host name or IP to connect to
  276.      * @param string  $user     username to log into the SMTP server with
  277.      * @param string  $pass     password to log into the SMTP server with
  278.      * @param integer $length 
  279.      *
  280.      * @return handle $stream file handle resource to SMTP stream
  281.      */
  282.     function initStream($message$length=0$host=''$port=''$user=''$pass=''{
  283.         return $stream;
  284.     }
  285.  
  286.     /**
  287.      * function getBCC - reserved for extended functionality
  288.      *
  289.      * This function is not yet implemented.
  290.      * Reserved for extended functionality.
  291.      *
  292.      */
  293.     function getBCC({
  294.         return false;
  295.     }
  296.  
  297.     /**
  298.      * function prepareMIME_Header - creates the mime header
  299.      *
  300.      * @param Message $message  Message object to act on
  301.      * @param string  $boundary mime boundary from fn MimeBoundary
  302.      *
  303.      * @return string $header properly formatted mime header
  304.      */
  305.     function prepareMIME_Header($message$boundary{
  306.         $mime_header $message->mime_header;
  307.         $rn="\r\n";
  308.         $header array();
  309.  
  310.         $contenttype 'Content-Type: '$mime_header->type0 .'/'.
  311.                         $mime_header->type1;
  312.         if (count($message->entities)) {
  313.             $contenttype .= ';' 'boundary="'.$boundary.'"';
  314.         }
  315.         if (isset($mime_header->parameters['name'])) {
  316.             $contenttype .= '; name="'.
  317.             encodeHeader($mime_header->parameters['name'])'"';
  318.         }
  319.         if (isset($mime_header->parameters['charset'])) {
  320.             $charset $mime_header->parameters['charset'];
  321.             $contenttype .= '; charset="'.
  322.             encodeHeader($charset)'"';
  323.         }
  324.  
  325.         $header[$contenttype $rn;
  326.         if ($mime_header->description{
  327.             $header['Content-Description: ' $mime_header->description $rn;
  328.         }
  329.         if ($mime_header->encoding{
  330.             $header['Content-Transfer-Encoding: ' $mime_header->encoding $rn;
  331.         else {
  332.             if ($mime_header->type0 == 'text' || $mime_header->type0 == 'message'{
  333.                 $header['Content-Transfer-Encoding: 8bit' .  $rn;
  334.             else {
  335.                 $header['Content-Transfer-Encoding: base64' .  $rn;
  336.             }
  337.         }
  338.         if ($mime_header->id{
  339.             $header['Content-ID: ' $mime_header->id $rn;
  340.         }
  341.         if ($mime_header->disposition{
  342.             $disposition $mime_header->disposition;
  343.             $contentdisp 'Content-Disposition: ' $disposition->name;
  344.             if ($disposition->getProperty('filename')) {
  345.                 $contentdisp .= '; filename="'.
  346.                 encodeHeader($disposition->getProperty('filename'))'"';
  347.             }
  348.             $header[$contentdisp $rn;
  349.         }
  350.         if ($mime_header->md5{
  351.             $header['Content-MD5: ' $mime_header->md5 $rn;
  352.         }
  353.         if ($mime_header->language{
  354.             $header['Content-Language: ' $mime_header->language $rn;
  355.         }
  356.  
  357.         $cnt count($header);
  358.         $hdr_s '';
  359.         for ($i $i $cnt $i++)    {
  360.             $hdr_s .= $this->foldLine($header[$i]78,str_pad('',4));
  361.         }
  362.         $header $hdr_s;
  363.         $header .= $rn/* One blank line to separate mimeheader and body-entity */
  364.         return $header;
  365.     }
  366.  
  367.     /**
  368.      * function prepareRFC822_Header - prepares the RFC822 header string from Rfc822Header object(s)
  369.      *
  370.      * This function takes the Rfc822Header object(s) and formats them
  371.      * into the RFC822Header string to send to the SMTP server as part
  372.      * of the SMTP message.
  373.      *
  374.      * @param Rfc822Header  $rfc822_header 
  375.      * @param Rfc822Header  $reply_rfc822_header 
  376.      * @param integer      &$raw_length length of the message
  377.      *
  378.      * @return string $header
  379.      */
  380.     function prepareRFC822_Header($rfc822_header$reply_rfc822_header&$raw_length{
  381.         global $domain$version$username$encode_header_key
  382.                $edit_identity$hide_auth_header;
  383.  
  384.         /* if server var SERVER_NAME not available, use $domain */
  385.         if(!sqGetGlobalVar('SERVER_NAME'$SERVER_NAMESQ_SERVER)) {
  386.             $SERVER_NAME $domain;
  387.         }
  388.  
  389.         sqGetGlobalVar('REMOTE_ADDR'$REMOTE_ADDRSQ_SERVER);
  390.         sqGetGlobalVar('REMOTE_PORT'$REMOTE_PORTSQ_SERVER);
  391.         sqGetGlobalVar('REMOTE_HOST'$REMOTE_HOSTSQ_SERVER);
  392.         sqGetGlobalVar('HTTP_VIA',    $HTTP_VIA,    SQ_SERVER);
  393.         sqGetGlobalVar('HTTP_X_FORWARDED_FOR'$HTTP_X_FORWARDED_FORSQ_SERVER);
  394.  
  395.         $rn "\r\n";
  396.  
  397.         /* This creates an RFC 822 date */
  398.         $date date('D, j M Y H:i:s 'time()) $this->timezone();
  399.         /* Create a message-id */
  400.         $message_id '<' $REMOTE_PORT '.';
  401.         if (isset($encode_header_key&& trim($encode_header_key)!=''{
  402.             // use encrypted form of remote address
  403.             $message_id.= OneTimePadEncrypt($this->ip2hex($REMOTE_ADDR),base64_encode($encode_header_key));
  404.         else {
  405.             $message_id.= $REMOTE_ADDR;
  406.         }
  407.         $message_id .= '.' time('.squirrel@' $SERVER_NAME .'>';
  408.         /* Make an RFC822 Received: line */
  409.         if (isset($REMOTE_HOST)) {
  410.             $received_from "$REMOTE_HOST ([$REMOTE_ADDR])";
  411.         else {
  412.             $received_from $REMOTE_ADDR;
  413.         }
  414.         if (isset($HTTP_VIA|| isset ($HTTP_X_FORWARDED_FOR)) {
  415.             if (!isset($HTTP_X_FORWARDED_FOR|| $HTTP_X_FORWARDED_FOR == ''{
  416.                 $HTTP_X_FORWARDED_FOR 'unknown';
  417.             }
  418.             $received_from .= " (proxying for $HTTP_X_FORWARDED_FOR)";
  419.         }
  420.         $header array();
  421.  
  422.         /**
  423.          * SquirrelMail header
  424.          *
  425.          * This Received: header provides information that allows to track
  426.          * user and machine that was used to send email. Don't remove it
  427.          * unless you understand all possible forging issues or your
  428.          * webmail installation does not prevent changes in user's email address.
  429.          * See SquirrelMail bug tracker #847107 for more details about it.
  430.          *
  431.          * Add $hide_squirrelmail_header as a candidate for config_local.php
  432.          * to allow completely hiding SquirrelMail participation in message
  433.          * processing; This is dangerous, especially if users can modify their 
  434.          * account information, as it makes mapping a sent message back to the
  435.          * original sender almost impossible.
  436.          */
  437.         $show_sm_header defined('hide_squirrelmail_header'hide_squirrelmail_header );
  438.  
  439.         if $show_sm_header {
  440.           if (isset($encode_header_key&&
  441.             trim($encode_header_key)!=''{
  442.             // use encoded headers, if encryption key is set and not empty
  443.             $header['X-Squirrel-UserHash: '.OneTimePadEncrypt($username,base64_encode($encode_header_key)).$rn;
  444.             $header['X-Squirrel-FromHash: '.OneTimePadEncrypt($this->ip2hex($REMOTE_ADDR),base64_encode($encode_header_key)).$rn;
  445.             if (isset($HTTP_X_FORWARDED_FOR))
  446.                 $header['X-Squirrel-ProxyHash:'.OneTimePadEncrypt($this->ip2hex($HTTP_X_FORWARDED_FOR),base64_encode($encode_header_key)).$rn;
  447.           else {
  448.             // use default received headers
  449.             $header["Receivedfrom $received_from$rn;
  450.             if ($edit_identity || isset($hide_auth_header|| $hide_auth_header)
  451.                 $header["        (SquirrelMail authenticated user $username)$rn;
  452.             $header["        by $SERVER_NAME with HTTP;$rn;
  453.             $header["        $date$rn;
  454.           }
  455.         }
  456.  
  457.         /* Insert the rest of the header fields */
  458.         $header['Message-ID: '$message_id $rn;
  459.         if (is_object($reply_rfc822_header&&
  460.             isset($reply_rfc822_header->message_id&&
  461.             $reply_rfc822_header->message_id{