Source for file compose.php

Documentation is available at compose.php

  1. <?php
  2. /**
  3.  * compose.php
  4.  *
  5.  * This code sends a mail.
  6.  *
  7.  * There are 4 modes of operation:
  8.  *    - Start new mail
  9.  *    - Add an attachment
  10.  *    - Send mail
  11.  *    - Save As Draft
  12.  *
  13.  * @copyright 1999-2020 The SquirrelMail Project Team
  14.  * @license http://opensource.org/licenses/gpl-license.php GNU Public License
  15.  * @version $Id: compose.php 14845 2020-01-07 08:09:34Z pdontthink $
  16.  * @package squirrelmail
  17.  */
  18.  
  19. /** This is the compose page */
  20. define('PAGE_NAME''compose');
  21.  
  22. /**
  23.  * Include the SquirrelMail initialization file.
  24.  */
  25. require('../include/init.php');
  26.  
  27. /* If email_address not set and admin wants us to ask user for it,
  28.  * redirect to options page. */
  29. if $ask_user_info && getPref($data_dir$username,'email_address'== "" {
  30.     header("Location: " get_location("/options.php?optpage=personal");
  31.     exit;
  32. }
  33.  
  34. /* SquirrelMail required files. */
  35. require_once(SM_PATH 'functions/imap_general.php');
  36. require_once(SM_PATH 'functions/imap_messages.php');
  37. require_once(SM_PATH 'functions/date.php');
  38. require_once(SM_PATH 'functions/mime.php');
  39. require_once(SM_PATH 'functions/compose.php');
  40. require_once(SM_PATH 'class/deliver/Deliver.class.php');
  41. require_once(SM_PATH 'functions/addressbook.php');
  42. require_once(SM_PATH 'functions/forms.php');
  43. require_once(SM_PATH 'functions/identity.php');
  44. global $imap_stream_options// in case not defined in config
  45.  
  46. /* --------------------- Get globals ------------------------------------- */
  47.  
  48. /** SESSION VARS */
  49. sqgetGlobalVar('delimiter'$delimiter,     SQ_SESSION);
  50.  
  51. sqgetGlobalVar('delayed_errors',  $delayed_errors,  SQ_SESSION);
  52. sqgetGlobalVar('composesession',    $composesession,    SQ_SESSION);
  53. sqgetGlobalVar('compose_messages',  $compose_messages,  SQ_SESSION);
  54.  
  55. // compose_messages only useful in SESSION when a forward-as-attachment
  56. // has been preconstructed for us and passed in via that mechanism; once
  57. // we have it, we can clear it from the SESSION
  58. // -- No, this is useful in other scenarios, too -- removing:
  59. // sqsession_unregister('compose_messages');
  60.  
  61. // Turn on delayed error handling in case we wind up redirecting below
  62. $oErrorHandler->setDelayedErrors(true);
  63.  
  64. /** SESSION/POST/GET VARS */
  65. sqgetGlobalVar('send_button_count'$send_button_countSQ_POST1SQ_TYPE_INT);
  66. for ($i 1$i <= $send_button_count$i++)
  67.    if (sqgetGlobalVar('send' $i$sendSQ_POST)) break;
  68. // Send can only be achieved by setting $_POST var. If Send = true then
  69. // retrieve other form fields from $_POST
  70. if (isset($send&& $send{
  71.     $SQ_GLOBAL SQ_POST;
  72. else {
  73.     $SQ_GLOBAL SQ_FORM;
  74. }
  75. sqgetGlobalVar('session',$session$SQ_GLOBAL);
  76. sqgetGlobalVar('mailbox',$mailbox$SQ_GLOBAL);
  77. sqgetGlobalVar('identity',$orig_identity$SQ_GLOBAL);
  78. if(!sqgetGlobalVar('identity',$identity$SQ_GLOBAL)) {
  79.     $identity=0;
  80. }
  81. sqgetGlobalVar('send_to',$send_to$SQ_GLOBAL);
  82. sqgetGlobalVar('send_to_cc',$send_to_cc$SQ_GLOBAL);
  83. sqgetGlobalVar('send_to_bcc',$send_to_bcc$SQ_GLOBAL);
  84. sqgetGlobalVar('subject',$subject$SQ_GLOBAL);
  85. sqgetGlobalVar('body',$body$SQ_GLOBAL);
  86. sqgetGlobalVar('mailprio',$mailprio$SQ_GLOBAL);
  87. sqgetGlobalVar('request_mdn',$request_mdn$SQ_GLOBAL);
  88. sqgetGlobalVar('request_dr',$request_dr$SQ_GLOBAL);
  89. sqgetGlobalVar('html_addr_search',$html_addr_search$SQ_GLOBAL);
  90. sqgetGlobalVar('mail_sent',$mail_sent$SQ_GLOBAL);
  91. sqgetGlobalVar('passed_id',$passed_id$SQ_GLOBALNULLSQ_TYPE_BIGINT);
  92. sqgetGlobalVar('passed_ent_id',$passed_ent_id$SQ_GLOBAL);
  93. sqgetGlobalVar('fwduid',$fwduid$SQ_GLOBAL'');
  94.  
  95. sqgetGlobalVar('attach',$attachSQ_POST);
  96. sqgetGlobalVar('draft',$draftSQ_POST);
  97. sqgetGlobalVar('draft_id',$draft_id$SQ_GLOBAL);
  98. sqgetGlobalVar('ent_num',$ent_num$SQ_GLOBAL);
  99. sqgetGlobalVar('saved_draft',$saved_draftSQ_FORM);
  100.  
  101. if sqgetGlobalVar('delete_draft',$delete_draft) ) {
  102.     $delete_draft = (int)$delete_draft;
  103. }
  104.  
  105. if sqgetGlobalVar('startMessage',$startMessage) ) {
  106.     $startMessage = (int)$startMessage;
  107. else {
  108.     $startMessage 1;
  109. }
  110.  
  111.  
  112. /** POST VARS */
  113. sqgetGlobalVar('sigappend',             $sigappend,                 SQ_POST);
  114. sqgetGlobalVar('from_htmladdr_search',  $from_htmladdr_search,      SQ_POST);
  115. sqgetGlobalVar('addr_search_done',      $html_addr_search_done,     SQ_POST);
  116. sqgetGlobalVar('addr_search_cancel',    $html_addr_search_cancel,   SQ_POST);
  117. sqgetGlobalVar('send_to_search',        $send_to_search,            SQ_POST);
  118. sqgetGlobalVar('do_delete',             $do_delete,                 SQ_POST);
  119. sqgetGlobalVar('delete',                $delete,                    SQ_POST);
  120. sqgetGlobalVar('attachments',           $attachments,               SQ_POST);
  121. if sqgetGlobalVar('return'$tempSQ_POST) ) {
  122.     $html_addr_search_done 'Use Addresses';
  123. }
  124.  
  125. /** GET VARS */
  126. if sqgetGlobalVar('account'$temp,  SQ_GET) ) {
  127.     $iAccount = (int) $temp;
  128. else {
  129.     $iAccount 0;
  130. }
  131.  
  132.  
  133. /** get smaction */
  134. if !sqgetGlobalVar('smaction',$action) )
  135. {
  136.     if sqgetGlobalVar('smaction_reply',$tmp) )      $action 'reply';
  137.     if sqgetGlobalVar('smaction_reply_all',$tmp) )  $action 'reply_all';
  138.     if sqgetGlobalVar('smaction_forward',$tmp) )    $action 'forward';
  139.     if sqgetGlobalVar('smaction_attache',$tmp) )    $action 'forward_as_attachment';
  140.     if sqgetGlobalVar('smaction_draft',$tmp) )      $action 'draft';
  141.     if sqgetGlobalVar('smaction_edit_new',$tmp) )   $action 'edit_as_new';
  142. }
  143.  
  144. sqgetGlobalVar('smtoken'$submitted_token$SQ_GLOBAL'');
  145.  
  146. /**
  147.  * Here we decode the data passed in from mailto.php.
  148.  */
  149. if sqgetGlobalVar('mailtodata'$mailtodataSQ_GET) ) {
  150.     $trtable array('to'       => 'send_to',
  151.                  'cc'           => 'send_to_cc',
  152.                  'bcc'          => 'send_to_bcc',
  153.                  'body'         => 'body',
  154.                  'subject'      => 'subject');
  155.     $mtdata unserialize($mailtodata);
  156.  
  157.     foreach ($trtable as $f => $t{
  158.         if !empty($mtdata[$f]) ) {
  159.             $$t $mtdata[$f];
  160.         }
  161.     }
  162.     unset($mailtodata,$mtdata$trtable);
  163. }
  164.  
  165. /* Location (For HTTP 1.1 header("Location: ...") redirects) */
  166. $location get_location();
  167. /* Identities (fetch only once) */
  168. $idents get_identities();
  169.  
  170. /* --------------------- Specific Functions ------------------------------ */
  171.  
  172. function replyAllString($header{
  173.     global $include_self_reply_all$idents;
  174.     $excl_ar array();
  175.     /**
  176.      * 1) Remove the addresses we'll be sending the message 'to'
  177.      */
  178.     if (isset($header->reply_to&& is_array($header->reply_to&& count($header->reply_to)) {
  179.         $excl_ar $header->getAddr_a('reply_to');
  180.     else if (is_object($header->reply_to)) /* unneccesarry, just for failsafe purpose */
  181.         $excl_ar $header->getAddr_a('reply_to');
  182.     else {
  183.         $excl_ar $header->getAddr_a('from');
  184.     }
  185.     /**
  186.      * 2) Remove our identities from the CC list (they still can be in the
  187.      * TO list) only if $include_self_reply_all is turned off
  188.      */
  189.     if (!$include_self_reply_all{
  190.         foreach($idents as $id{
  191.             $excl_ar[strtolower(trim($id['email_address']))'';
  192.         }
  193.     }
  194.  
  195.     /**
  196.      * 3) get the addresses.
  197.      */
  198.     $url_replytoall_ar $header->getAddr_a(array('to','cc')$excl_ar);
  199.  
  200.     /**
  201.      * 4) generate the string.
  202.      */
  203.     $url_replytoallcc '';
  204.     foreach$url_replytoall_ar as $email => $personal{
  205.         if ($personal{
  206.             // always quote personal name (can't just quote it if
  207.             // it contains a comma separator, since it might still
  208.             // be encoded)
  209.             $url_replytoallcc .= ", \"$personal\" <$email>";
  210.         else {
  211.             $url_replytoallcc .= ', '$email;
  212.         }
  213.     }
  214.     $url_replytoallcc substr($url_replytoallcc,2);
  215.  
  216.     return $url_replytoallcc;
  217. }
  218.  
  219. /**
  220.  * creates top line in reply citations
  221.  *
  222.  * Line style depends on user preferences.
  223.  * $orig_date argument is available only from 1.4.3 and 1.5.1 version.
  224.  * @param object $orig_from From: header object.
  225.  * @param integer $orig_date email's timestamp
  226.  * @return string reply citation
  227.  */
  228. function getReplyCitation($orig_from$orig_date{
  229.     global $reply_citation_style$reply_citation_start$reply_citation_end;
  230.  
  231.     if (!is_object($orig_from)) {
  232.         $sOrig_from '';
  233.     else {
  234.         $sOrig_from decodeHeader($orig_from->getAddress(false),false,false,true);
  235.     }
  236.  
  237.     /* First, return an empty string when no citation style selected. */
  238.     if (($reply_citation_style == ''|| ($reply_citation_style == 'none')) {
  239.         return '';
  240.     }
  241.  
  242.     /* Make sure our final value isn't an empty string. */
  243.     if ($sOrig_from == ''{
  244.         return '';
  245.     }
  246.  
  247.     /* Otherwise, try to select the desired citation style. */
  248.     switch ($reply_citation_style{
  249.     case 'author_said':
  250.         // i18n: %s is for author's name
  251.         $full_reply_citation sprintf(_("%s wrote:"),$sOrig_from);
  252.         break;
  253.     case 'quote_who':
  254.         // TODO: the words "quote" and "who" are translated in 1.4.x so why not here?  This isn't a real HTML tag...
  255.         $start '<quote who="';
  256.         $end   '">';
  257.         $full_reply_citation $start $sOrig_from $end;
  258.         break;
  259.     case 'date_time_author':
  260.         // i18n:
  261.         // The first %s is for date string, the second %s is for author's name.
  262.         // The date uses formating from "D, F j, Y g:i a" and "D, F j, Y H:i"
  263.         // translations.
  264.         // Example string:
  265.         // "On Sat, December 24, 2004 23:59, Santa wrote:"
  266.         // If you have to put author's name in front of date string, check comments about
  267.         // argument swapping at http://php.net/sprintf
  268.         $full_reply_citation sprintf(_("On %s, %s wrote:")getLongDateString($orig_date)$sOrig_from);
  269.         break;
  270.     case 'user-defined':
  271.         $start $reply_citation_start .
  272.             ($reply_citation_start == '' '' ' ');
  273.         $end   $reply_citation_end;
  274.         $full_reply_citation $start $sOrig_from $end;
  275.         break;
  276.     default:
  277.         return '';
  278.     }
  279.  
  280.     /* Add line feed and return the citation string. */
  281.     return ($full_reply_citation "\n");
  282. }
  283.  
  284. /**
  285.  * Creates header fields in forwarded email body
  286.  *
  287.  * $default_charset global must be set correctly before you call this function.
  288.  * @param object $orig_header 
  289.  * @return $string 
  290.  */
  291. function getforwardHeader($orig_header{
  292.     global $editor_size$default_charset;
  293.  
  294.     // using own strlen function in order to detect correct string length
  295.     $display array_("Subject"=> sq_strlen(_("Subject"),$default_charset),
  296.             _("From")    => sq_strlen(_("From"),$default_charset),
  297.             _("Date")    => sq_strlen(_("Date"),$default_charset),
  298.             _("To")      => sq_strlen(_("To"),$default_charset),
  299.             _("Cc")      => sq_strlen(_("Cc"),$default_charset) );
  300.     $maxsize max($display);
  301.     $indent str_pad('',$maxsize+2);
  302.     foreach($display as $key => $val{
  303.         $display[$key$key .': 'str_pad(''$maxsize $val);
  304.     }
  305.     $from decodeHeader($orig_header->getAddr_s('from',"\n$indent"),false,false,true);
  306.     $from str_replace('&nbsp;',' ',$from);
  307.     $to decodeHeader($orig_header->getAddr_s('to',"\n$indent"),false,false,true);
  308.     $to str_replace('&nbsp;',' ',$to);
  309.     $subject decodeHeader($orig_header->subject,false,false,true);
  310.     $subject str_replace('&nbsp;',' ',$subject);
  311.  
  312.     // using own str_pad function in order to create correct string pad
  313.     $bodyTop =  sq_str_pad(' '._("Original Message").' ',$editor_size -2,'-',STR_PAD_BOTH,$default_charset.
  314.         "\n"$display[_("Subject")$subject "\n" .
  315.         $display[_("From")$from "\n" .
  316.         $display[_("Date")getLongDateString$orig_header->date$orig_header->date_unparsed )"\n" .
  317.         $display[_("To")$to "\n";
  318.     if ($orig_header->cc != array(&& $orig_header->cc !=''{
  319.         $cc decodeHeader($orig_header->getAddr_s('cc',"\n$indent"),false,false,true);
  320.         $cc str_replace('&nbsp;',' ',$cc);
  321.         $bodyTop .= $display[_("Cc").$cc "\n";
  322.     }
  323.     $bodyTop .= str_pad(''$editor_size -'-'.
  324.         "\n\n";
  325.     return $bodyTop;
  326. }
  327. /* ----------------------------------------------------------------------- */
  328.  
  329. /*
  330.  * If the session is expired during a post this restores the compose session
  331.  * vars.
  332.  */
  333. $session_expired false;
  334. if (sqsession_is_registered('session_expired_post')) {
  335.     sqgetGlobalVar('session_expired_post'$session_expired_postSQ_SESSION);
  336.     /*
  337.      * extra check for username so we don't display previous post data from
  338.      * another user during this session.
  339.      */
  340.     if (!empty($session_expired_post['username']
  341.      && $session_expired_post['username'== $username{
  342.         // these are the vars that we can set from the expired composed session
  343.         $compo_var_list array ('send_to''send_to_cc''body',
  344.             'startMessage''passed_body''use_signature''signature',
  345.             'subject''newmail''send_to_bcc''passed_id''mailbox'
  346.             'from_htmladdr_search''identity''draft_id''delete_draft'
  347.             'mailprio''edit_as_new''attachments''composesession'
  348.             'request_mdn''request_dr''fwduid');
  349.  
  350.         foreach ($compo_var_list as $var{
  351.             if isset($session_expired_post[$var]&& !isset($$var) ) {
  352.                 $$var $session_expired_post[$var];
  353.             }
  354.         }
  355.  
  356.         if (!empty($attachments))
  357.             $attachments unserialize(urldecode($attachments));
  358.  
  359.         sqsession_register($composesession,'composesession');
  360.  
  361.         if (isset($send)) {
  362.             unset($send);
  363.         }
  364.         $session_expired true;
  365.     }
  366.     unset($session_expired_post);
  367.     sqsession_unregister('session_expired_post');
  368.     if (!isset($mailbox)) {
  369.         $mailbox '';
  370.     }
  371.     if ($compose_new_win == '1'{
  372.         compose_Header($color$mailbox);
  373.     else {
  374.         $sHeaderJs (isset($sHeaderJs)) $sHeaderJs '';
  375.         if (strpos($action'reply'!== false && $reply_focus{
  376.             $sOnload 'checkForm(\''.$replyfocus.'\');';
  377.         else {
  378.             $sOnload 'checkForm();';
  379.         }
  380.         displayPageHeader($color$mailbox,$sHeaderJs,$sOnload);
  381.     }
  382.     showInputForm($sessionfalse);
  383.     exit();
  384. }
  385.  
  386. if (!isset($composesession)) {
  387.     $composesession 0;
  388.     sqsession_register(0,'composesession');
  389. else {
  390.     $composesession = (int)$composesession;
  391. }
  392.  
  393. if (!isset($session|| (isset($newmessage&& $newmessage)) {
  394.     sqsession_unregister('composesession');
  395.     $session "$composesession+1;
  396.     $composesession $session;
  397.     sqsession_register($composesession,'composesession');
  398. }
  399. if (!empty($compose_messages[$session])) {
  400.     $composeMessage $compose_messages[$session];
  401. else {
  402.     $composeMessage new Message();
  403.     $rfc822_header new Rfc822Header();
  404.     $composeMessage->rfc822_header $rfc822_header;
  405.     $composeMessage->reply_rfc822_header '';
  406. }
  407.  
  408. // re-add attachments that were already in this message
  409. // FIXME: note that technically this is very bad form -
  410. // should never directly manipulate an object like this
  411. if (!empty($attachments)) {
  412.     $attachments unserialize(urldecode($attachments));
  413.     if (!empty($attachments&& is_array($attachments)) {
  414.         // sanitize the "att_local_name" since it is user-supplied and used to access the file system
  415.         // it must be alpha-numeric and 32 characters long (see the use of GenerateRandomString() below)
  416.         foreach ($attachments as $i => $attachment{
  417.             if (empty($attachment->att_local_name|| strlen($attachment->att_local_name!== 32{
  418.                 unset($attachments[$i]);
  419.                 continue;
  420.             }
  421.             // probably marginal difference between (ctype_alnum + function_exists) and preg_match
  422.             if (function_exists('ctype_alnum')) {
  423.                 if (!ctype_alnum($attachment->att_local_name))
  424.                     unset($attachments[$i]);
  425.             }
  426.             else if (preg_match('/[^0-9a-zA-Z]/'$attachment->att_local_name))
  427.                 unset($attachments[$i]);
  428.         }
  429.         if (!empty($attachments))
  430.             $composeMessage->entities $attachments;
  431.     }
  432. }
  433.  
  434. if (empty($mailbox)) {
  435.     $mailbox 'INBOX';
  436. }
  437.  
  438. if ($draft{
  439.  
  440.     // validate security token
  441.     //
  442.     sm_validate_security_token($submitted_token-1TRUE);
  443.  
  444.     /*
  445.      * Set $default_charset to correspond with the user's selection
  446.      * of language interface.
  447.      */
  448.     set_my_charset();
  449.     if (deliverMessage($composeMessagetrue)) {
  450.         showInputForm($session);
  451.         exit();
  452.     else {
  453.         $draft_message _("Draft Email Saved");
  454.         /* If this is a resumed draft, then delete the original */
  455.         if(isset($delete_draft)) {
  456.             $imap_stream sqimap_login($usernamefalse$imapServerAddress$imapPortfalse$imap_stream_options);
  457.             sqimap_mailbox_select($imap_stream$draft_folder);
  458.             // force bypass_trash=true because message should be saved when deliverMessage() returns true.
  459.             // in current implementation of sqimap_msgs_list_flag() single message id can
  460.             // be submitted as string. docs state that it should be array.
  461.             sqimap_msgs_list_delete($imap_stream$draft_folder$delete_drafttrue);
  462.             if ($auto_expunge{
  463.                 sqimap_mailbox_expunge($imap_stream$draft_foldertrue);
  464.             }
  465.             sqimap_logout($imap_stream);
  466.         }
  467.  
  468.         $oErrorHandler->saveDelayedErrors();
  469.         session_write_close();
  470.  
  471.         if ($compose_new_win == '1'{
  472.             if !isset($pageheader_sent|| !$pageheader_sent {
  473.                 header("Location: $location/compose.php?saved_draft=yes&session=$composesession");
  474.             else {
  475. //FIXME: DON'T ECHO HTML FROM CORE!
  476.                 echo '   <br><br><div style="text-align: center;"><a href="' $location
  477.                     . '/compose.php?saved_sent=yes&amp;session=' $composesession '">'
  478.                     . _("Return"'</a></div>';
  479.             }
  480.             exit();
  481.         else {
  482.             if !isset($pageheader_sent|| !$pageheader_sent {
  483.                 header("Location: $location/right_main.php?mailbox=urlencode($draft_folder.
  484.                    "&startMessage=1&note=".urlencode($draft_message));
  485.             else {
  486. //FIXME: DON'T ECHO HTML FROM CORE!
  487.                 echo '   <br><br><div style="text-align: center;"><a href="' $location
  488.                     . '/right_main.php?mailbox=' urlencode($draft_folder)
  489.                     . '&amp;startMessage=1&amp;note=' urlencode($draft_message.'">'
  490.                     . _("Return"'</a></div>';
  491.             }
  492.             exit();
  493.         }
  494.     }
  495. }
  496.  
  497. if ($send{
  498.  
  499.     // validate security token
  500.     //
  501.     sm_validate_security_token($submitted_token-1TRUE);
  502.  
  503.     if (isset($_FILES['attachfile']&&
  504.             $_FILES['attachfile']['tmp_name'&&
  505.             $_FILES['attachfile']['tmp_name'!= 'none'{
  506.         $AttachFailure saveAttachedFiles($session);
  507.     }
  508.     
  509.     if (checkInput(false&& !isset($AttachFailure)) {
  510.         if ($mailbox == "All Folders"{
  511.             /* We entered compose via the search results page */
  512.             $mailbox 'INBOX'/* Send 'em to INBOX, that's safe enough */
  513.         }
  514.         $urlMailbox urlencode($mailbox);
  515.         if (isset($passed_id)) {
  516.             $passed_id 0;
  517.         }
  518.         /**
  519.          * Set $default_charset to correspond with the user's selection
  520.          * of language interface.
  521.          */
  522.         set_my_charset();
  523.         /**
  524.          * This is to change all newlines to \n
  525.          * We'll change them to \r\n later (in the sendMessage function)
  526.          */
  527.         $body str_replace("\r\n""\n"$body);
  528.         $body str_replace("\r""\n"$body);
  529.  
  530.         /**
  531.          * Rewrap $body so that no line is bigger than $editor_size
  532.          */
  533.         $body explode("\n"$body);
  534.         $newBody '';
  535.         foreach ($body as $line{
  536.             if$line <> '-- ' {
  537.                 $line rtrim($line);
  538.             }
  539.             if (sq_strlen($line$default_charset<= $editor_size 1{
  540.                 $newBody .= $line "\n";
  541.             else {
  542.                 sqWordWrap($line$editor_size$default_charset);
  543.                 $newBody .= $line "\n";
  544.  
  545.             }
  546.  
  547.         }
  548.         $body $newBody;
  549.  
  550.         $Result deliverMessage($composeMessage);
  551.  
  552.         if ($Result)
  553.             $mail_sent 'yes';
  554.         else
  555.             $mail_sent 'no';
  556.  
  557.         // NOTE: this hook changed in 1.5.2 from sending $Result and
  558.         //       $composeMessage as args #2 and #3 to being in an array
  559.         //       under arg #2
  560.         $temp array(&$Result&$composeMessage&$mail_sent);
  561.         do_hook('compose_send_after'$temp);
  562.         if ($Result{
  563.             showInputForm($session);
  564.             exit();
  565.         }
  566.  
  567.         /* if it is resumed draft, delete draft message */
  568.         if isset($delete_draft)) {
  569.             $imap_stream sqimap_login($usernamefalse$imapServerAddress$imapPortfalse$imap_stream_options);
  570.             sqimap_mailbox_select($imap_stream$draft_folder);
  571.             // bypass_trash=true because message should be saved when deliverMessage() returns true.
  572.             // in current implementation of sqimap_msgs_list_flag() single message id can
  573.             // be submitted as string. docs state that it should be array.
  574.             sqimap_msgs_list_delete($imap_stream$draft_folder$delete_drafttrue);
  575.             if ($auto_expunge{
  576.                 sqimap_mailbox_expunge($imap_stream$draft_foldertrue);
  577.             }
  578.             sqimap_logout($imap_stream);
  579.         }
  580.         /*
  581.          * Store the error array in the session because they will be lost on a redirect
  582.          */
  583.         $oErrorHandler->saveDelayedErrors();
  584.         session_write_close();
  585.  
  586.         if ($compose_new_win == '1'{
  587.             if !isset($pageheader_sent|| !$pageheader_sent {
  588.                 header("Location: $location/compose.php?mail_sent=$mail_sent");
  589.             else {
  590. //FIXME: DON'T ECHO HTML FROM CORE!
  591.                 echo '   <br><br><div style="text-align: center;"><a href="' $location
  592.                     . '/compose.php?mail_sent=$mail_sent">'
  593.                     . _("Return"'</a></div>';
  594.             }
  595.             exit();
  596.         else {
  597.             if !isset($pageheader_sent|| !$pageheader_sent {
  598.                 global $return_to_message_after_reply;
  599.                 if (($action === 'reply' || $action === 'reply_all' || $action === 'forward' || $action === 'forward_as_attachment')
  600.                  && $return_to_message_after_reply && $passed_id)
  601.                     header("Location: $location/read_body.php?passed_id=$passed_id&mailbox=$urlMailbox".
  602.                             "&startMessage=$startMessage&mail_sent=$mail_sent");
  603.                 else
  604.                     header("Location: $location/right_main.php?mailbox=$urlMailbox".
  605.                             "&startMessage=$startMessage&mail_sent=$mail_sent");
  606.             else {
  607. //FIXME: DON'T ECHO HTML FROM CORE!
  608.                 echo '   <br><br><div style="text-align: center;"><a href="' $location
  609.                     . "/right_main.php?mailbox=$urlMailbox"
  610.                     . "&amp;startMessage=$startMessage&amp;mail_sent=$mail_sent\">"
  611.                     . _("Return"'</a></div>';
  612.             }
  613.             exit();
  614.         }
  615.     else {
  616.         if ($compose_new_win == '1'{
  617.             compose_Header($color$mailbox);
  618.         }
  619.         else {
  620.             displayPageHeader($color$mailbox);
  621.         }
  622.         if (isset($AttachFailure)) {
  623.             plain_error_message(_("Could not move/copy file. File not attached"),
  624.                     $color);
  625.         }
  626.         checkInput(true);
  627.         showInputForm($session);
  628.         /* sqimap_logout($imapConnection); */
  629.     }
  630. elseif (isset($html_addr_search_done)) {
  631.  
  632.     // validate security token
  633.     //
  634.     sm_validate_security_token($submitted_token-1TRUE);
  635.  
  636.     if ($compose_new_win == '1'{
  637.         compose_Header($color$mailbox);
  638.     }
  639.     else {
  640.         displayPageHeader($color$mailbox);
  641.     }
  642.  
  643.     if (isset($send_to_search&& is_array($send_to_search)) {
  644.         foreach ($send_to_search as $k => $v{
  645.             if (substr($k01== 'T'{
  646.                 if ($send_to{
  647.                     $send_to .= ', ';
  648.                 }
  649.                 $send_to .= $v;
  650.             }
  651.             elseif (substr($k01== 'C'{
  652.                 if ($send_to_cc{
  653.                     $send_to_cc .= ', ';
  654.                 }
  655.                 $send_to_cc .= $v;
  656.             }
  657.             elseif (substr($k01== 'B'{
  658.                 if ($send_to_bcc{
  659.                     $send_to_bcc .= ', ';
  660.                 }
  661.                 $send_to_bcc .= $v;
  662.             }
  663.         }
  664.     }
  665.     showInputForm($session);
  666. elseif (isset($html_addr_search&& !isset($html_addr_search_cancel)) {
  667.     if (isset($_FILES['attachfile']&&
  668.             $_FILES['attachfile']['tmp_name'&&
  669.             $_FILES['attachfile']['tmp_name'!= 'none'{
  670.         if(saveAttachedFiles($session)) {
  671.             plain_error_message(_("Could not move/copy file. File not attached"));
  672.         }
  673.     }
  674.     /*
  675.      * I am using an include so as to elminiate an extra unnecessary
  676.      * click.  If you can think of a better way, please implement it.
  677.      */
  678.     include_once('./addrbook_search_html.php');
  679. elseif (isset($attach)) {
  680.  
  681.     // validate security token
  682.     //
  683.     sm_validate_security_token($submitted_token-1TRUE);
  684.  
  685.     if ($compose_new_win == '1'{
  686.         compose_Header($color$mailbox);
  687.     else {
  688.         displayPageHeader($color$mailbox);
  689.     }
  690.     if (saveAttachedFiles($session)) {
  691.         plain_error_message(_("Could not move/copy file. File not attached"));
  692.     }
  693.     showInputForm($session);
  694. }
  695. elseif (isset($sigappend)) {
  696.  
  697.     // validate security token
  698.     //
  699.     sm_validate_security_token($submitted_token-1TRUE);
  700.  
  701.     $signature $idents[$identity]['signature'];
  702.  
  703.     $body .= "\n\n".($prefix_sig==true"-- \n":'').$signature;
  704.     if ($compose_new_win == '1'{
  705.         compose_Header($color$mailbox);
  706.     else {
  707.         displayPageHeader($color$mailbox);
  708.     }
  709.     showInputForm($session);
  710. elseif (isset($do_delete)) {
  711.  
  712.     // validate security token
  713.     //
  714.     sm_validate_security_token($submitted_token-1TRUE);
  715.  
  716.     if ($compose_new_win == '1'{
  717.         compose_Header($color$mailbox);
  718.     else {
  719.         displayPageHeader($color$mailbox);
  720.     }
  721.  
  722.     if (isset($delete&& is_array($delete)) {
  723.         foreach($delete as $index{
  724.             if (!empty($composeMessage->entities&& isset($composeMessage->entities[$index])) {
  725.                 $composeMessage->entities[$index]->purgeAttachments();
  726.                 // FIXME: one person reported that unset() didn't do anything at all here, so this is a work-around... but it triggers PHP notices if the unset() doesn't work, which should be fixed... but bigger question is if unset() doesn't work here, what about everywhere else? Anyway, uncomment this if you think you need it
  727.                 //$composeMessage->entities[$index] = NULL;
  728.                 unset ($composeMessage->entities[$index]);
  729.             }
  730.         }
  731.         $new_entities array();
  732.         foreach ($composeMessage->entities as $entity{
  733.             $new_entities[$entity;
  734.         }
  735.         $composeMessage->entities $new_entities;
  736.     }
  737.     showInputForm($session);
  738. else {
  739.     /*
  740.      * This handles the default case as well as the error case
  741.      * (they had the same code) --> if (isset($smtpErrors))
  742.      */
  743.  
  744.     if ($compose_new_win == '1'{
  745.         compose_Header($color$mailbox);
  746.     else {
  747.         displayPageHeader($color$mailbox);
  748.     }
  749.  
  750.     $newmail true;
  751.  
  752.     if (!isset($passed_ent_id)) {
  753.         $passed_ent_id '';
  754.     }
  755.     if (!isset($passed_id)) {
  756.         $passed_id '';
  757.     }
  758.     if (!isset($mailbox)) {
  759.         $mailbox '';
  760.     }
  761.     if (!isset($action)) {
  762.         $action '';
  763.     }
  764.  
  765.     $values newMail($mailbox,$passed_id,$passed_ent_id$action$session);
  766.  
  767.     // forward as attachment - subject is in the message in session
  768.     //
  769.     if ($action == 'forward_as_attachment' && empty($values['subject']))
  770.         $subject $composeMessage->rfc822_header->subject;
  771.  
  772.     /* in case the origin is not read_body.php */
  773.     if (isset($send_to)) {
  774.         $values['send_to'$send_to;
  775.     }
  776.     if (isset($send_to_cc)) {
  777.         $values['send_to_cc'$send_to_cc;
  778.     }
  779.     if (isset($send_to_bcc)) {
  780.         $values['send_to_bcc'$send_to_bcc;
  781.     }
  782.     if (isset($subject)) {
  783.         $values['subject'$subject;
  784.     }
  785.     if (isset($mailprio)) {
  786.         $values['mailprio'$mailprio;
  787.     }
  788.     if (isset($orig_identity)) {
  789.         $values['identity'$orig_identity;
  790.     }
  791.     showInputForm($session$values);
  792. }
  793.  
  794. exit();
  795.  
  796. /**************** Only function definitions go below *************/
  797.  
  798. function getforwardSubject($subject)
  799. {
  800.     if ((substr(strtolower($subject)04!= 'fwd:'&&
  801.             (substr(strtolower($subject)05!= '[fwd:'&&
  802.             (substr(strtolower($subject)06!= '[ fwd:')) {
  803.         $subject '[Fwd: ' $subject ']';
  804.     }
  805.     return $subject;
  806. }
  807.  
  808. /* This function is used when not sending or adding attachments */
  809. function newMail ($mailbox=''$passed_id=''$passed_ent_id=''$action=''$session=''{
  810.     global $editor_size$default_use_priority$body$idents,
  811.         $use_signature$data_dir$username,
  812.         $key$imapServerAddress$imapPort$imap_stream_options,
  813.         $composeMessage$body_quote$request_mdn$request_dr,
  814.         $mdn_user_support$languages$squirrelmail_language,
  815.         $default_charset$do_not_reply_to_self$compose_messages;
  816.  
  817.     /*
  818.      * Set $default_charset to correspond with the user's selection
  819.      * of language interface. $default_charset global is not correct,
  820.      * if message is composed in new window.
  821.      */
  822.     set_my_charset();
  823.  
  824.     $send_to $send_to_cc $send_to_bcc $subject $identity '';
  825.     $mailprio 3;
  826.  
  827.     if ($passed_id{
  828.         $imapConnection sqimap_login($usernamefalse$imapServerAddress,
  829.                 $imapPort0$imap_stream_options);
  830.  
  831.         sqimap_mailbox_select($imapConnection$mailbox);
  832.         $message sqimap_get_message($imapConnection$passed_id$mailbox);
  833.  
  834.         $body '';
  835.         if ($passed_ent_id{
  836.             /* redefine the messsage in case of message/rfc822 */
  837.             $message $message->getEntity($passed_ent_id);
  838.             /* message is an entity which contains the envelope and type0=message
  839.              * and type1=rfc822. The actual entities are childs from
  840.              * $message->entities[0]. That's where the encoding and is located
  841.              */
  842.  
  843.             $entities $message->entities[0]->findDisplayEntity
  844.                 (array()$alt_order array('text/plain'));
  845.             if (!count($entities)) {
  846.                 $entities $message->entities[0]->findDisplayEntity
  847.                     (array()$alt_order array('text/plain','text/html'));
  848.             }
  849.             $orig_header $message->rfc822_header/* here is the envelope located */
  850.             /* redefine the message for picking up the attachments */
  851.             $message $message->entities[0];
  852.  
  853.         else {
  854.             $entities $message->findDisplayEntity (array()$alt_order array('text/plain'));
  855.             if (!count($entities)) {
  856.                 $entities $message->findDisplayEntity (array()$alt_order array('text/plain','text/html'));
  857.             }
  858.             $orig_header $message->rfc822_header;
  859.         }
  860.  
  861.         $type0 $message->type0;
  862.         $type1 $message->type1;
  863.         foreach ($entities as $ent{
  864.             $msg $message->getEntity($ent);
  865.             $type0 $msg->type0;
  866.             $type1 $msg->type1;
  867.             $unencoded_bodypart mime_fetch_body($imapConnection$passed_id$ent);
  868.             $body_part_entity $message->getEntity($ent);
  869.             $bodypart decodeBody($unencoded_bodypart,
  870.                     $body_part_entity->header->encoding);
  871.             if ($type1 == 'html'{
  872.                 $bodypart str_replace("\n"' '$bodypart);
  873.                 $bodypart preg_replace(array('/<\/?p>/i','/<div><\/div>/i','/<br\s*(\/)*>/i','/<\/?div>/i')"\n"$bodypart);
  874.                 $bodypart str_replace(array('&nbsp;','&gt;','&lt;'),array(' ','>','<'),$bodypart);
  875.                 $bodypart strip_tags($bodypart);
  876.             }
  877.             if (isset($languages[$squirrelmail_language]['XTRA_CODE']&&
  878.                     function_exists($languages[$squirrelmail_language]['XTRA_CODE''_decode')) {
  879.                 if (mb_detect_encoding($bodypart!= 'ASCII'{
  880.                     $bodypart call_user_func($languages[$squirrelmail_language]['XTRA_CODE''_decode'$bodypart);
  881.                 }
  882.             }
  883.  
  884.             // charset encoding in compose form stuff
  885.             if (isset($body_part_entity->header->parameters['charset'])) {
  886.                 $actual $body_part_entity->header->parameters['charset'];
  887.             else {
  888.                 $actual 'us-ascii';
  889.             }
  890.  
  891.             if $actual && is_conversion_safe($actual&& $actual != $default_charset){
  892.                 $bodypart charset_convert($actual,$bodypart,$default_charset,false);
  893.             }
  894.             // end of charset encoding in compose
  895.  
  896.             $body .= $bodypart;
  897.         }
  898.         if ($default_use_priority{
  899.             $mailprio substr($orig_header->priority,0,1);
  900.             if (!$mailprio{
  901.                 $mailprio 3;
  902.             }
  903.         else {
  904.             $mailprio '';
  905.         }
  906.  
  907.         $from_o $orig_header->from;
  908.         if (is_array($from_o)) {
  909.             if (isset($from_o[0])) {
  910.                 $from_o $from_o[0];
  911.             }
  912.         }
  913.         if (is_object($from_o)) {
  914.             $orig_from $from_o->getAddress();
  915.         else {
  916.             $orig_from '';
  917.         }
  918.  
  919.         $identities array();
  920.         if (count($idents1{
  921.             foreach($idents as $nr=>$data{
  922.                 $enc_from_name '"'.$data['full_name'].'" <'$data['email_address'].'>';
  923.                 $identities[$enc_from_name;
  924.             }
  925.  
  926.             $identity_match $orig_header->findAddress($identities);
  927.             if ($identity_match !== FALSE{
  928.                 $identity $identity_match;
  929.             }
  930.         }
  931.  
  932.         switch ($action{
  933.             case ('draft'):
  934.                 $use_signature FALSE;
  935.                 $composeMessage->rfc822_header $orig_header;
  936.                 $send_to decodeHeader($orig_header->getAddr_s('to'),false,false,true);
  937.                 $send_to_cc decodeHeader($orig_header->getAddr_s('cc'),false,false,true);
  938.                 $send_to_bcc decodeHeader($orig_header->getAddr_s('bcc'),false,false,true);
  939.                 $send_from $orig_header->getAddr_s('from');
  940.                 $send_from_parts new AddressStructure();
  941.                 $send_from_parts $orig_header->parseAddress($send_from);
  942.                 $send_from_add $send_from_parts->mailbox '@' $send_from_parts->host;
  943.                 $identity find_identity(array($send_from_add));
  944.                 $subject decodeHeader($orig_header->subject,false,false,true);
  945.  
  946.                 // Remember the receipt settings
  947.                 $request_mdn $mdn_user_support && !empty($orig_header->dnt'1' '0';
  948.                 $request_dr $mdn_user_support && !empty($orig_header->drnt'1' '0';
  949.  
  950.                 /* remember the references and in-reply-to headers in case of an reply */
  951. //FIXME: it would be better to fiddle with headers inside of the message object or possibly when delivering the message to its destination (drafts folder?); is this possible?
  952.                 $composeMessage->rfc822_header->more_headers['References'$orig_header->references;
  953.                 $composeMessage->rfc822_header->more_headers['In-Reply-To'$orig_header->in_reply_to;
  954.                 // rewrap the body to clean up quotations and line lengths
  955.                 sqBodyWrap($body$editor_size);
  956.                 $composeMessage getAttachments($message$composeMessage$passed_id$entities$imapConnection);
  957.                 if (!empty($orig_header->x_sm_flag_reply))
  958.                     $composeMessage->rfc822_header->more_headers['X-SM-Flag-Reply'$orig_header->x_sm_flag_reply;
  959. //TODO: completely unclear if should be using $compose_session instead of $session below
  960.                 $compose_messages[$session$composeMessage;
  961.                 sqsession_register($compose_messages,'compose_messages');
  962.                 break;
  963.             case ('edit_as_new'):
  964.                 $send_to decodeHeader($orig_header->getAddr_s('to'),false,false,true);
  965.                 $send_to_cc decodeHeader($orig_header->getAddr_s('cc'),false,false,true);
  966.                 $send_to_bcc decodeHeader($orig_header->getAddr_s('bcc'),false,false,true);
  967.                 $subject decodeHeader($orig_header->subject,false,false,true);
  968.                 $mailprio $orig_header->priority;
  969.                 $orig_from '';
  970.                 $composeMessage getAttachments($message$composeMessage$passed_id$entities$imapConnection);
  971.                 // rewrap the body to clean up quotations and line lengths
  972.                 sqBodyWrap($body$editor_size);
  973.                 break;
  974.             case ('forward'):
  975.                 $send_to '';
  976.                 $subject getforwardSubject(decodeHeader($orig_header->subject,false,false,true));
  977.                 $body getforwardHeader($orig_header$body;
  978.                 // the logic for calling sqUnWordWrap here would be to allow the browser to wrap the lines
  979.                 // forwarded message text should be as undisturbed as possible, so commenting out this call
  980.                 // sqUnWordWrap($body);
  981.                 $composeMessage getAttachments($message$composeMessage$passed_id$entities$imapConnection);
  982.  
  983.                 //add a blank line after the forward headers
  984.                 $body "\n" $body;
  985.                 break;
  986.             case ('forward_as_attachment'):
  987.                 $subject getforwardSubject(decodeHeader($orig_header->subject,false,false,true));
  988.                 $composeMessage getMessage_RFC822_Attachment($message$composeMessage$passed_id$passed_ent_id$imapConnection);
  989.                 $subject decodeHeader($orig_header->subject,false,false,true);
  990.                 $subject str_replace('"'"'"$subject);
  991.                 $subject trim($subject);
  992.                 if (substr(strtolower($subject)04!= 'fwd:'{
  993.                     $subject 'Fwd: ' $subject;
  994.                 }
  995.                 $body '';
  996.                 break;
  997.             case ('reply_all'):
  998.                 if(isset($orig_header->mail_followup_to&& $orig_header->mail_followup_to{
  999.                     $send_to $orig_header->getAddr_s('mail_followup_to');
  1000.                 else {
  1001.                     $send_to_cc replyAllString($orig_header);
  1002.                     $send_to_cc decodeHeader($send_to_cc,false,false,true);
  1003.                     $send_to_cc str_replace('""''"'$send_to_cc);
  1004.                 }
  1005.             case ('reply'):
  1006.                 // skip this if send_to was already set right above here
  1007.                 if(!$send_to{
  1008.                     $send_to $orig_header->reply_to;
  1009.                     if (is_array($send_to&& count($send_to)) {
  1010.                         $send_to $orig_header->getAddr_s('reply_to'','FALSETRUE);
  1011.                     else if (is_object($send_to)) /* unneccesarry, just for failsafe purpose */
  1012.                         $send_to $orig_header->getAddr_s('reply_to'','FALSETRUE);
  1013.                     else {
  1014.                         $send_to $orig_header->getAddr_s('from'','FALSETRUE);
  1015.                     }
  1016.                 }
  1017.                 $send_to decodeHeader($send_to,false,false,true);
  1018.                 $send_to str_replace('""''"'$send_to);
  1019.  
  1020.  
  1021.                 // If user doesn't want replies to her own messages
  1022.                 // going back to herself (instead send again to the
  1023.                 // original recipient of the message being replied to),
  1024.                 // then iterate through identities, checking if the TO
  1025.                 // field is one of them (if the reply is to ourselves)
  1026.                 //
  1027.                 // Note we don't bother if the original message doesn't
  1028.                 // have anything in the TO field itself (because that's
  1029.                 // what we use if we change the recipient to be that of
  1030.                 // the previous message)
  1031.                 //
  1032.                 if ($do_not_reply_to_self && !empty($orig_header->to)) {
  1033.  
  1034.                     $orig_to '';
  1035.  
  1036.                     foreach($idents as $id{
  1037.  
  1038.                         if (!empty($id['email_address'])
  1039.                          && strpos($send_to$id['email_address']!== FALSE{
  1040.  
  1041.                             // if this is a reply-all, the original recipient
  1042.                             // is already in the CC field, so we can just blank
  1043.                             // the recipient (TO field) (as long as the CC field
  1044.                             // isn't empty that is)... but then move the CC into
  1045.                             // the TO, so TO isn't empty
  1046.                             //
  1047.                             if ($action == 'reply_all' && !empty($send_to_cc)) {
  1048.                                 $orig_to $send_to_cc;
  1049.                                 $send_to_cc '';
  1050.                                 break;
  1051.                             }
  1052.  
  1053.                             $orig_to $orig_header->to;
  1054.                             if (is_array($orig_to&& count($orig_to)) {
  1055.                                 $orig_to $orig_header->getAddr_s('to'','FALSETRUE);
  1056.                             else if (is_object($orig_to)) /* unneccesarry, just for failsafe purpose */
  1057.                                 $orig_to $orig_header->getAddr_s('to'','FALSETRUE);
  1058.                             else {
  1059.                                 $orig_to '';
  1060.                             }
  1061.                             $orig_to decodeHeader($orig_to,false,false,true);
  1062.                             $orig_to str_replace('""''"'$orig_to);
  1063.  
  1064.                             break;
  1065.                         }
  1066.                     }
  1067.  
  1068.                     // if the reply was addressed back to ourselves,
  1069.                     // we will send it to the TO of the previous message
  1070.                     //
  1071.                     if (!empty($orig_to)) {
  1072.  
  1073.                         $send_to $orig_to;
  1074.  
  1075.                         // in this case, we also want to reset the FROM
  1076.                         // identity as well (it should match the original
  1077.                         // *FROM* header instead of TO or CC)
  1078.                         //
  1079.                         if (count($idents1{
  1080.                             $identity '';
  1081.                             foreach($idents as $i => $id{
  1082.                                 if (!empty($id['email_address'])
  1083.                                  && strpos($orig_from$id['email_address']!== FALSE{
  1084.                                     $identity $i;
  1085.                                     break;
  1086.                                 }
  1087.                             }
  1088.                         }
  1089.  
  1090.                     }
  1091.  
  1092.                 }
  1093.  
  1094.  
  1095.                 $subject decodeHeader($orig_header->subject,false,false,true);
  1096.                 $subject str_replace('"'"'"$subject);
  1097.                 $subject trim($subject);
  1098.                 if (substr(strtolower($subject)03!= 're:'{
  1099.                     $subject 'Re: ' $subject;
  1100.                 }
  1101.                 /* this corrects some wrapping/quoting problems on replies */
  1102.                 $rewrap_body explode("\n"$body);
  1103.                 $from (is_array($orig_header->from&& !empty($orig_header->from)) $orig_header->from[0$orig_header->from;
  1104.                 $body '';
  1105.                 $strip_sigs getPref($data_dir$username'strip_sigs');
  1106.                 foreach ($rewrap_body as $line{
  1107.                     if ($strip_sigs && rtrim($line"\r\n"== '-- '{
  1108.                         break;
  1109.                     }
  1110.                     if (preg_match("/^(>+)/"$line$matches)) {
  1111.                         $gt $matches[1];
  1112.                         $body .= $body_quote str_replace("\n""\n$body_quote$gt "rtrim($line)) ."\n";
  1113.                     else {
  1114.                         $body .= $body_quote (!empty($body_quote' ' ''str_replace("\n""\n$body_quote(!empty($body_quote' ' '')rtrim($line)) "\n";
  1115.                     }
  1116.                 }
  1117.  
  1118.                 //rewrap the body to clean up quotations and line lengths
  1119.                 $body sqBodyWrap ($body$editor_size);
  1120.  
  1121.                 $body getReplyCitation($from $orig_header->date$body;
  1122.                 $composeMessage->reply_rfc822_header $orig_header;
  1123.  
  1124.                 break;
  1125.             default:
  1126.                 break;
  1127.         }
  1128. //FIXME: we used to register $compose_messages in the session here, but not any more - so do we still need the session_write_close() and sqimap_logout() here?  We probably need the IMAP logout, but what about the session closure?
  1129.         session_write_close();
  1130.         sqimap_logout($imapConnection);
  1131.     }
  1132.     $ret array'send_to' => $send_to,
  1133.             'send_to_cc' => $send_to_cc,
  1134.             'send_to_bcc' => $send_to_bcc,
  1135.             'subject' => $subject,
  1136.             'mailprio' => $mailprio,
  1137.             'body' => $body,
  1138.             'identity' => $identity );
  1139.  
  1140.     return ($ret);
  1141. /* function newMail() */
  1142.  
  1143. /**
  1144.  * downloads attachments from original message, stores them in attachment directory and adds
  1145.  * them to composed message.
  1146.  * @param object $message 
  1147.  * @param object $composeMessage 
  1148.  * @param integer $passed_id 
  1149.  * @param mixed $entities 
  1150.  * @param mixed $imapConnection 
  1151.  * @return object 
  1152.  */
  1153. function getAttachments($message&$composeMessage$passed_id$entities$imapConnection{
  1154.     global $squirrelmail_language$languages$username$attachment_dir;
  1155.  
  1156.     if (!count($message->entities||
  1157.             ($message->type0 == 'message' && $message->type1 == 'rfc822')) {
  1158.         if !in_array($message->entity_id$entities&& $message->entity_id{
  1159.             switch ($message->type0{
  1160.                 case 'message':
  1161.                     if ($message->type1 == 'rfc822'{
  1162.                         $filename $message->rfc822_header->subject;
  1163.                         if ($filename == ""{
  1164.                             $filename "untitled-".$message->entity_id;
  1165.                         }
  1166.                         $filename .= '.eml';
  1167.                     else {
  1168.                         $filename $message->getFilename();
  1169.                     }
  1170.                     break;
  1171.                 default:
  1172.                     if (!$message->mime_header/* temporary hack */
  1173.                         $message->mime_header $message->header;
  1174.                     }
  1175.                     $filename $message->getFilename();
  1176.                     break;
  1177.             }
  1178. //FIXME: added three args to the following, so as to set the last one to TRUE, to mimick a fix in 1.4.21 (#2994865), but didn't test this (note that in 1.4.21, the 2nd and 3rd args are FALSE, but here in this code, they weren't being specified (thus defaulting to TRUE), so I don't know if that means this code is outdated and should have been changed to FALSE, FALSE or if this code is completely different and the addition of the TRUE for arg #4 is wrong
  1179.             $filename str_replace('&#32;'' 'decodeHeader($filenametruetruetrue));
  1180.             if (isset($languages[$squirrelmail_language]['XTRA_CODE']&&
  1181.                     function_exists($languages[$squirrelmail_language]['XTRA_CODE''_encode')) {
  1182.                 $filename =  call_user_func($languages[$squirrelmail_language]['XTRA_CODE''_encode'$filename);
  1183.             }
  1184.  
  1185.             $hashed_attachment_dir getHashedDir($username$attachment_dir);
  1186.             $localfilename sq_get_attach_tempfile();
  1187.             $message->att_local_name $localfilename;
  1188.  
  1189.             $composeMessage->initAttachment($message->type0.'/'.$message->type1,$filename,
  1190.                     $localfilename);
  1191.  
  1192.             /* Write Attachment to file */
  1193.             $fp fopen ($hashed_attachment_dir '/' $localfilename'wb');
  1194.             mime_print_body_lines ($imapConnection$passed_id$message->entity_id$message->header->encoding$fp);
  1195.             fclose ($fp);
  1196.         }
  1197.     else {
  1198.         for ($i=0$entCount=count($message->entities)$i<$entCount;$i++{
  1199.             $composeMessage=getAttachments($message->entities[$i]$composeMessage$passed_id$entities$imapConnection);
  1200.         }
  1201.     }
  1202.     return $composeMessage;
  1203. }
  1204.  
  1205. function getMessage_RFC822_Attachment($message$composeMessage$passed_id,
  1206.         $passed_ent_id=''$imapConnection{
  1207.     if (!$passed_ent_id{
  1208.         $body_a sqimap_run_command($imapConnection,
  1209.                 'FETCH '.$passed_id.' RFC822',
  1210.                 TRUE$response$readmessage,
  1211.                 TRUE);
  1212.     else {
  1213.         $body_a sqimap_run_command($imapConnection,
  1214.                 'FETCH '.$passed_id.' BODY['.$passed_ent_id.']',
  1215.                 TRUE$response$readmessageTRUE);
  1216.         $message $message->parent;
  1217.     }
  1218.     if ($response == 'OK'{
  1219.         $subject encodeHeader($message->rfc822_header->subject);
  1220.         array_shift($body_a);
  1221.         array_pop($body_a);
  1222.         $body implode(''$body_a"\r\n";
  1223.  
  1224.         global $username$attachment_dir;
  1225.         $hashed_attachment_dir getHashedDir($username$attachment_dir);
  1226.         $localfilename sq_get_attach_tempfile();
  1227.         $fp fopen($hashed_attachment_dir '/' $localfilename'wb');
  1228.         fwrite ($fp$body);
  1229.         fclose($fp);
  1230.         $composeMessage->initAttachment('message/rfc822',$subject.'.eml',
  1231.                 $localfilename);
  1232.     }
  1233.     return $composeMessage;
  1234. }
  1235.  
  1236. function showInputForm ($session$values=false{
  1237.     global $send_to$send_to_cc$send_to_bcc,
  1238.         $body$startMessage$action$attachments,
  1239.         $use_signature$signature$prefix_sig$session_expired,
  1240.         $editor_size$editor_height$subject$newmail,
  1241.         $use_javascript_addr_book$passed_id$mailbox$fwduid,
  1242.         $from_htmladdr_search$location_of_buttons$attachment_dir,
  1243.         $username$data_dir$identity$idents$delete_draft,
  1244.         $mailprio$compose_new_win$saved_draft$mail_sent$sig_first,
  1245.         $composeMessage$composesession$default_charset,
  1246.         $compose_onsubmit$oTemplate$oErrorHandler;
  1247.  
  1248.     if (checkForJavascript()) {
  1249.         $onfocus ' onfocus="alreadyFocused=true;"';
  1250.         $onfocus_array array('onfocus' => 'alreadyFocused=true;');
  1251.     }
  1252.     else {
  1253.         $onfocus '';
  1254.         $onfocus_array array();
  1255.     }
  1256.  
  1257.     if ($values{
  1258.         $send_to $values['send_to'];
  1259.         $send_to_cc $values['send_to_cc'];
  1260.         $send_to_bcc $values['send_to_bcc'];
  1261.         $subject $values['subject'];
  1262.         $mailprio $values['mailprio'];
  1263.         $body $values['body'];
  1264.         $identity = (int) $values['identity'];
  1265.     else {
  1266.         $send_to decodeHeader($send_totruefalse);
  1267.         $send_to_cc decodeHeader($send_to_cctruefalse);
  1268.         $send_to_bcc decodeHeader($send_to_bcctruefalse);
  1269.     }
  1270.  
  1271.     if ($use_javascript_addr_book{
  1272. //FIXME: NO HTML IN CORE!
  1273.         echo "\n"'<script type="text/javascript">'."\n<!--\n" .
  1274.             'function open_abook() { ' "\n" .
  1275.             '  var nwin = window.open("addrbook_popup.php","abookpopup",' .
  1276.             '"width=670,height=300,resizable=yes,scrollbars=yes");' "\n" .
  1277.             '  if((!nwin.opener) && (document.windows != null))' "\n" .
  1278.             '    nwin.opener = document.windows;' "\n" .
  1279.             "}\n" .
  1280.             "// -->\n</script>\n\n";
  1281.     }
  1282.  
  1283. //FIXME: NO HTML IN CORE!
  1284.     echo "\n" '<form name="compose" action="compose.php" method="post" ' .
  1285.         'enctype="multipart/form-data"';
  1286.  
  1287.     $compose_onsubmit array();
  1288.     global $null;
  1289.     do_hook('compose_form'$null);
  1290.  
  1291.     // Plugins that use compose_form hook can add an array entry
  1292.     // to the globally scoped $compose_onsubmit; we add them up
  1293.     // here and format the form tag's full onsubmit handler.
  1294.     // Each plugin should use "return false" if they need to
  1295.     // stop form submission but otherwise should NOT use "return
  1296.     // true" to give other plugins the chance to do what they need
  1297.     // to do; SquirrelMail itself will add the final "return true".
  1298.     // Onsubmit text is enclosed inside of double quotes, so plugins
  1299.     // need to quote accordingly.
  1300.     //
  1301.     // Also, plugin authors should try to retain compatibility with
  1302.     // the Compose Extras plugin by resetting its compose submit
  1303.     // counter when preventing form submit.  Use this code: 
  1304.     // if (your-code-here) { submit_count = 0; return false; }
  1305.     //
  1306.     if (checkForJavascript()) {
  1307.         if (empty($compose_onsubmit))
  1308.             $compose_onsubmit array();
  1309.         else if (!is_array($compose_onsubmit))
  1310.             $compose_onsubmit array($compose_onsubmit);
  1311.  
  1312.         $onsubmit_text '';
  1313.         foreach ($compose_onsubmit as $text{
  1314.             $text trim($text);
  1315.             if (!empty($text)) {
  1316.                 if (substr($text-1!= ';' && substr($text-1!= '}')
  1317.                     $text .= '; ';
  1318.                 $onsubmit_text .= $text;
  1319.             }
  1320.         }
  1321.  
  1322.         if (!empty($onsubmit_text))
  1323. //FIXME: DON'T ECHO HTML FROM CORE!
  1324.             echo ' onsubmit="' $onsubmit_text ' return true;"';
  1325.     }
  1326.  
  1327.  
  1328. //FIXME: NO HTML IN CORE!
  1329.     echo ">\n";
  1330.  
  1331. //FIXME: DON'T ECHO HTML FROM CORE!
  1332.     echo addHidden('smtoken'sm_generate_security_token());
  1333.  
  1334. //FIXME: DON'T ECHO HTML FROM CORE!
  1335.     echo addHidden('startMessage'$startMessage);
  1336.  
  1337.     if ($action == 'draft'{
  1338. //FIXME: DON'T ECHO HTML FROM CORE!
  1339.         echo addHidden('delete_draft'$passed_id);
  1340.     }
  1341.     if (isset($delete_draft)) {
  1342. //FIXME: DON'T ECHO HTML FROM CORE!
  1343.         echo addHidden('delete_draft'$delete_draft);
  1344.     }
  1345.     if (isset($session)) {
  1346. //FIXME: DON'T ECHO HTML FROM CORE!
  1347.         echo addHidden('session'$session);
  1348.     }
  1349.  
  1350.     if (isset($passed_id)) {
  1351. //FIXME: DON'T ECHO HTML FROM CORE!
  1352.         echo addHidden('passed_id'$passed_id);
  1353.     }
  1354.  
  1355.     if (isset($fwduid)) {
  1356. //FIXME: DON'T ECHO HTML FROM CORE!
  1357.         echo addHidden('fwduid'$fwduid);
  1358.     }
  1359.  
  1360.     if ($saved_draft == 'yes'{
  1361.         $oTemplate->assign('note'_("Your draft has been saved."));
  1362.         $oTemplate->display('note.tpl');
  1363.     }
  1364.     if ($mail_sent == 'yes'{
  1365.         $oTemplate->assign('note'_("Your mail has been sent."));
  1366.         $oTemplate->display('note.tpl');
  1367.     }
  1368.     if ($compose_new_win == '1'{
  1369.         $oTemplate->display('compose_newwin_close.tpl');
  1370.     }
  1371.  
  1372.     if ($location_of_buttons == 'top'{
  1373. //FIXME: DON'T ECHO HTML FROM CORE!
  1374.         showComposeButtonRow();
  1375.     }
  1376.  
  1377.     $identities array();
  1378.     if (count($idents1{
  1379.         reset($idents);
  1380.         foreach($idents as $id => $data{
  1381.             $identities[$id$data['full_name'].' &lt;'.$data['email_address'].'&gt;';
  1382.         }
  1383.     }
  1384.  
  1385.     $oTemplate->assign('identities'$identities);
  1386.     $oTemplate->assign('identity_def'$identity);
  1387.     $oTemplate->assign('input_onfocus''onfocus="'.join(' '$onfocus_array).'"');
  1388.  
  1389.     $oTemplate->assign('to'sm_encode_html_special_chars($send_to));
  1390.     $oTemplate->assign('cc'sm_encode_html_special_chars($send_to_cc));
  1391.     $oTemplate->assign('bcc'sm_encode_html_special_chars($send_to_bcc));
  1392.     $oTemplate->assign('subject'sm_encode_html_special_chars($subject));
  1393.  
  1394.     // access keys...
  1395.     //
  1396.     global $accesskey_compose_to$accesskey_compose_cc,
  1397.            $accesskey_compose_identity$accesskey_compose_bcc,
  1398.            $accesskey_compose_subject;
  1399.     $oTemplate->assign('accesskey_compose_identity'$accesskey_compose_identity);
  1400.     $oTemplate->assign('accesskey_compose_to'$accesskey_compose_to);
  1401.     $oTemplate->assign('accesskey_compose_cc'$accesskey_compose_cc);
  1402.     $oTemplate->assign('accesskey_compose_bcc'$accesskey_compose_bcc);
  1403.     $oTemplate->assign('accesskey_compose_subject'$accesskey_compose_subject);
  1404.  
  1405.     $oTemplate->display('compose_header.tpl');
  1406.  
  1407.     if ($location_of_buttons == 'between'{
  1408. //FIXME: DON'T ECHO HTML FROM CORE!
  1409.         showComposeButtonRow();
  1410.     }
  1411.  
  1412.     $body_str '';
  1413.     if ($use_signature == true && $newmail == true && !isset($from_htmladdr_search)) {
  1414.         $signature $idents[$identity]['signature'];
  1415.  
  1416.         if ($sig_first == '1'{
  1417.             /*
  1418.              * FIXME: test is specific to ja_JP translation implementation.
  1419.              * This test might apply incorrect conversion to other translations, but
  1420.              * use of 7bit iso-2022-jp charset in other translations might have other
  1421.              * issues too.
  1422.              */
  1423.             if ($default_charset == 'iso-2022-jp'{
  1424.                 $body_str "\n\n".($prefix_sig==true"-- \n":'').mb_convert_encoding($signature'EUC-JP');
  1425.             else {
  1426.                 $body_str "\n\n".($prefix_sig==true"-- \n":'').decodeHeader($signature,false,false);
  1427.             }
  1428.             $body_str .= "\n\n".sm_encode_html_special_chars(decodeHeader($body,false,false));
  1429.         else {
  1430.             $body_str "\n\n".sm_encode_html_special_chars(decodeHeader($body,false,false));
  1431.             // FIXME: test is specific to ja_JP translation implementation. See above comments.
  1432.             if ($default_charset == 'iso-2022-jp'{
  1433.                 $body_str .= "\n\n".($prefix_sig==true"-- \n":'').mb_convert_encoding($signature'EUC-JP');
  1434.             else {
  1435.                 $body_str .= "\n\n".($prefix_sig==true"-- \n":'').decodeHeader($signature,false,false);
  1436.             }
  1437.         }
  1438.     else {
  1439.         $body_str sm_encode_html_special_chars(decodeHeader($body,false,false));
  1440.     }
  1441.  
  1442.     $oTemplate->assign('editor_width'(int)$editor_size);
  1443.     $oTemplate->assign('editor_height'(int)$editor_height);
  1444.     $oTemplate->assign('input_onfocus''onfocus="'.join(' '$onfocus_array).'"');
  1445.     $oTemplate->assign('body'$body_str);
  1446.     $oTemplate->assign('show_bottom_send'$location_of_buttons!='bottom');
  1447.  
  1448.     // access keys...
  1449.     //
  1450.     global $accesskey_compose_body$accesskey_compose_send;
  1451.     $oTemplate->assign('accesskey_compose_body'$accesskey_compose_body);
  1452.     $oTemplate->assign('accesskey_compose_send'$accesskey_compose_send);
  1453.  
  1454.     $oTemplate->display ('compose_body.tpl');
  1455.  
  1456.     if ($location_of_buttons == 'bottom'{
  1457. //FIXME: DON'T ECHO HTML FROM CORE!
  1458.         showComposeButtonRow();
  1459.     }
  1460.  
  1461.     // composeMessage can be empty when coming from a restored session
  1462.     if (is_object($composeMessage&& $composeMessage->entities)
  1463.         $attach_array $composeMessage->entities;
  1464.     if ($session_expired && !empty($attachments&& is_array($attachments))
  1465.         $attach_array $attachments;
  1466.  
  1467.     /* This code is for attachments */
  1468.     if ((bool) ini_get('file_uploads')) {
  1469.  
  1470.         /* Calculate the max size for an uploaded file.
  1471.          * This is advisory for the user because we can't actually prevent
  1472.          * people to upload too large files. */
  1473.         $sizes array();
  1474.         /* php.ini vars which influence the max for uploads */
  1475.         $configvars array('post_max_size''memory_limit''upload_max_filesize');
  1476.         foreach($configvars as $var{
  1477.             /* skip 0 or empty values, and -1 which means 'unlimited' */
  1478.             if$size getByteSize(ini_get($var)) ) {
  1479.                 if $size != '-1' {
  1480.                     $sizes[$size;
  1481.                 }
  1482.             }
  1483.         }
  1484.  
  1485.         $attach array();
  1486.         global $username$attachment_dir;
  1487.         $hashed_attachment_dir getHashedDir($username$attachment_dir);
  1488.         if (!empty($attach_array)) {
  1489.             foreach ($attach_array as $key => $attachment{
  1490.                 $attached_file $attachment->att_local_name;
  1491.                 if ($attachment->att_local_name || $attachment->body_part{
  1492.                     $attached_filename decodeHeader($attachment->mime_header->getParameter('name'));
  1493.                     $type $attachment->mime_header->type0.'/'.
  1494.                         $attachment->mime_header->type1;
  1495.  
  1496.                     $a array();
  1497.                     $a['Key'$key;
  1498.                     $a['FileName'$attached_filename;
  1499.                     $a['ContentType'$type;
  1500.                     $a['Size'filesize($hashed_attachment_dir '/' $attached_file);
  1501.                     $attach[$key$a;
  1502.                 }
  1503.             }
  1504.         }
  1505.  
  1506.         $max min($sizes);
  1507.         $oTemplate->assign('max_file_size'empty($max? -$max);
  1508.         $oTemplate->assign('attachments'$attach);
  1509.  
  1510.         // access keys...
  1511.         //
  1512.         global $accesskey_compose_attach_browse$accesskey_compose_attach,
  1513.                $accesskey_compose_delete_attach;
  1514.         $oTemplate->assign('accesskey_compose_attach_browse'$accesskey_compose_attach_browse);
  1515.         $oTemplate->assign('accesskey_compose_attach'$accesskey_compose_attach);
  1516.         $oTemplate->assign('accesskey_compose_delete_attach'$accesskey_compose_delete_attach);
  1517.  
  1518.         $oTemplate->display('compose_attachments.tpl');
  1519.     // End of file_uploads if-block
  1520.     /* End of attachment code */
  1521.  
  1522.     $oTemplate->assign('username'$username);
  1523.     $oTemplate->assign('smaction'$action);
  1524.     $oTemplate->assign('mailbox'$mailbox);
  1525.     sqgetGlobalVar('QUERY_STRING'$queryStringSQ_SERVER);
  1526.     $oTemplate->assign('querystring'$queryString);
  1527.     $oTemplate->assign('composesession'$composesession);
  1528.     $oTemplate->assign('send_button_count'unique_widget_name('send'TRUE));
  1529.     if (!empty($attach_array))
  1530.         $oTemplate->assign('attachments'urlencode(serialize($attach_array)));
  1531.  
  1532.     $aUserNotices array();
  1533.  
  1534.     // File uploads are off, so we didn't show that part of the form.
  1535.     // To avoid bogus bug reports, tell the user why. 
  1536.     if (!(bool) ini_get('file_uploads')) {
  1537.         $aUserNotices[_("Because PHP file uploads are turned off, you can not attach files to this message. Please see your system administrator for details.");
  1538.     }
  1539.  
  1540.     $oTemplate->assign('user_notices'$aUserNotices);
  1541.  
  1542.     $oTemplate->display('compose_form_close.tpl');
  1543.  
  1544.     if ($compose_new_win=='1'{
  1545.         $oTemplate->display('compose_newwin_close.tpl');
  1546.     }
  1547.  
  1548.     $oErrorHandler->setDelayedErrors(false);
  1549.     $oTemplate->display('footer.tpl');
  1550. }
  1551.  
  1552.  
  1553. function showComposeButtonRow({
  1554.     global $use_javascript_addr_book$save_as_draft,
  1555.         $default_use_priority$mailprio$default_use_mdn,
  1556.         $request_mdn$request_dr,
  1557.         $data_dir$username;
  1558.  
  1559.     global $oTemplate$buffer_hook;
  1560.  
  1561.     if ($default_use_priority{
  1562.         $priorities array('1'=>_("High")'3'=>_("Normal")'5'=>_("Low"));
  1563.         $priority = isset($mailprio$mailprio 3;
  1564.     else {
  1565.         $priorities array();
  1566.         $priority NULL;
  1567.     }
  1568.  
  1569.     $mdn_user_support=getPref($data_dir$username'mdn_user_support',$default_use_mdn);
  1570.  
  1571.     $address_book_button_attribs array();
  1572.     global $accesskey_compose_addresses;
  1573.     if ($accesskey_compose_addresses != 'NONE')
  1574.         $address_book_button_attribs['accesskey'$accesskey_compose_addresses;
  1575.     if ($use_javascript_addr_book && checkForJavascript()) {
  1576.         $addr_book addButton(_("Addresses"),
  1577.                                null,
  1578.                                array_merge($address_book_button_attribsarray('onclick' => 'javascript:open_abook();')));
  1579.     else {
  1580.         $addr_book addSubmit(_("Addresses")'html_addr_search'$address_book_button_attribs);
  1581.     }
  1582.  
  1583.     $oTemplate->assign('allow_priority'$default_use_priority==1);
  1584.     $oTemplate->assign('priority_list'$priorities);
  1585.     $oTemplate->assign('current_priority'$priority);
  1586.  
  1587.     $oTemplate->assign('notifications_enabled'$mdn_user_support==1);
  1588.     $oTemplate->assign('read_receipt'$request_mdn=='1');
  1589.     $oTemplate->assign('delivery_receipt'$request_dr=='1');
  1590.  
  1591.     $oTemplate->assign('drafts_enabled'$save_as_draft);
  1592.     $oTemplate->assign('address_book_button'$addr_book);
  1593.  
  1594.     // access keys...
  1595.     //
  1596.     global $accesskey_compose_priority$accesskey_compose_on_read,
  1597.            $accesskey_compose_on_delivery$accesskey_compose_signature,
  1598.            $accesskey_compose_save_draft$accesskey_compose_send;
  1599.     $oTemplate->assign('accesskey_compose_priority'$accesskey_compose_priority);
  1600.     $oTemplate->assign('accesskey_compose_on_read'$accesskey_compose_on_read);
  1601.     $oTemplate->assign('accesskey_compose_on_delivery'$accesskey_compose_on_delivery);
  1602.     $oTemplate->assign('accesskey_compose_signature'$accesskey_compose_signature);
  1603.     $oTemplate->assign('accesskey_compose_save_draft'$accesskey_compose_save_draft);
  1604.     $oTemplate->assign('accesskey_compose_send'$accesskey_compose_send);
  1605.  
  1606.     $oTemplate->display('compose_buttons.tpl');
  1607. }
  1608.  
  1609. function checkInput ($show{
  1610.     /*
  1611.      * I implemented the $show variable because the error messages
  1612.      * were getting sent before the page header.  So, I check once
  1613.      * using $show=false, and then when i'm ready to display the error
  1614.      * message, show=true
  1615.      */
  1616.     global $send_to$send_to_cc$send_to_bcc;
  1617.  
  1618.     $send_to trim($send_to);
  1619.     $send_to_cc trim($send_to_cc);
  1620.     $send_to_bcc trim($send_to_bcc);
  1621.     if (empty($send_to&& empty($send_to_cc&& empty($send_to_bcc)) {
  1622.         if ($show{
  1623.             plain_error_message(_("You have not filled in the \"To:\" field."));
  1624.         }
  1625.         return false;
  1626.     }
  1627.     return true;
  1628. /* function checkInput() */
  1629.  
  1630.  
  1631. /* True if FAILURE */
  1632. function saveAttachedFiles($session{
  1633.     global $composeMessage$username$attachment_dir;
  1634.  
  1635.     /* get out of here if no file was attached at all */
  1636.     if (is_uploaded_file($_FILES['attachfile']['tmp_name']) ) {
  1637.         return true;
  1638.     }
  1639.  
  1640.     $hashed_attachment_dir getHashedDir($username$attachment_dir);
  1641.     $localfilename sq_get_attach_tempfile();
  1642.     $fullpath $hashed_attachment_dir '/' $localfilename;
  1643.  
  1644.     // m_u_f works better with restricted PHP installs (safe_mode, open_basedir),
  1645.     // if that doesn't work, try a simple rename.
  1646.     if (!sq_call_function_suppress_errors('move_uploaded_file'array($_FILES['attachfile']['tmp_name']$fullpath))) {
  1647.         if (!sq_call_function_suppress_errors('rename'array($_FILES['attachfile']['tmp_name']$fullpath))) {
  1648.             return true;
  1649.         }
  1650.     }
  1651.     $type strtolower($_FILES['attachfile']['type']);
  1652.     $name $_FILES['attachfile']['name'];
  1653.     $composeMessage->initAttachment($type$name$localfilename);
  1654. }
  1655.  
  1656. /**
  1657.   * Parse strings such as "8M" and "2k" into their corresponding size in bytes
  1658.   *
  1659.   * NOTE: This function only recognizes the suffixes "K", "M" and "G"
  1660.   *       and will probably break very easily if the given size is in
  1661.   *       some completely different format.
  1662.   *
  1663.   * @param string $ini_size The input string to be converted
  1664.   *
  1665.   * @return mixed Boolean FALSE if something went wrong (the value passed in
  1666.   *                was empty?, the suffix was not recognized?), otherwise, the
  1667.   *                converted size in bytes (just the number (as an integer),
  1668.   *                no unit identifier included)
  1669.   *
  1670.   */
  1671. function getByteSize($ini_size{
  1672.  
  1673.     if(!$ini_size{
  1674.         return FALSE;
  1675.     }
  1676.  
  1677.     $ini_size trim($ini_size);
  1678.  
  1679.     // if there's some kind of letter at the end of the string we need to multiply.
  1680.     if(!is_numeric(substr($ini_size-1))) {
  1681.  
  1682.         switch(strtoupper(substr($ini_size-1))) {
  1683.             case 'G':
  1684.                 $bytesize 1073741824;
  1685.                 break;
  1686.             case 'M':
  1687.                 $bytesize 1048576;
  1688.                 break;
  1689.             case 'K':
  1690.                 $bytesize 1024;
  1691.                 break;
  1692.              default:
  1693.                 return FALSE;
  1694.         }
  1695.  
  1696.         return ($bytesize * (int)substr($ini_size0-1));
  1697.     }
  1698.  
  1699.     return $ini_size;
  1700. }
  1701.  
  1702.  
  1703. /**
  1704.  * temporary function to make use of the deliver class.
  1705.  * In the future the responsible backend should be automaticly loaded
  1706.  * and conf.pl should show a list of available backends.
  1707.  * The message also should be constructed by the message class.
  1708.  *
  1709.  * @param object $composeMessage The message being sent.  Please note
  1710.  *                                that it is passed by reference and
  1711.  *                                will be returned modified, with additional
  1712.  *                                headers, such as Message-ID, Date, In-Reply-To,
  1713.  *                                References, and so forth.
  1714.  *
  1715.  * @return boolean FALSE if delivery failed, or some non-FALSE value
  1716.  *                  upon success.
  1717.  *
  1718.  */
  1719. function deliverMessage(&$composeMessage$draft=false{
  1720.     global $send_to$send_to_cc$send_to_bcc$mailprio$subject$body,
  1721.         $username$identity$idents$data_dir$compose_messages$session,
  1722.         $request_mdn$request_dr$default_charset$useSendmail,
  1723.         $domain$action$default_move_to_sent$move_to_sent,
  1724.         $imapServerAddress$imapPort$imap_stream_options$sent_folder$key;
  1725.  
  1726.     $rfc822_header $composeMessage->rfc822_header;
  1727.  
  1728.     $abook addressbook_init(falsetrue);
  1729.     $rfc822_header->to $rfc822_header->parseAddress($send_to,truearray()''$domainarray(&$abook,'lookup'));
  1730.     $rfc822_header->cc $rfc822_header->parseAddress($send_to_cc,true,array()'',$domainarray(&$abook,'lookup'));
  1731.     $rfc822_header->bcc $rfc822_header->parseAddress($send_to_bcc,truearray()'',$domainarray(&$abook,'lookup'));
  1732.     $rfc822_header->priority $mailprio;
  1733.     $rfc822_header->subject $subject;
  1734.  
  1735.     $special_encoding='';
  1736.     if (strtolower($default_charset== 'iso-2022-jp'{
  1737.         if (mb_detect_encoding($body== 'ASCII'{
  1738.             $special_encoding '8bit';
  1739.         else {
  1740.             $body mb_convert_encoding($body'JIS');
  1741.             $special_encoding '7bit';
  1742.         }
  1743.     }
  1744.     $composeMessage->setBody($body);
  1745.  
  1746.     $reply_to '';
  1747.     $reply_to  $idents[$identity]['reply_to'];
  1748.     if ($reply_to && strpos($reply_to'@'=== FALSE)
  1749.         $reply_to .= '@' $domain;
  1750.     
  1751.     $from_addr build_from_header($identity);
  1752.     $rfc822_header->from $rfc822_header->parseAddress($from_addr,true);
  1753.     if ($reply_to{
  1754.         $rfc822_header->reply_to $rfc822_header->parseAddress($reply_to,true);
  1755.     }
  1756.     /* Receipt: On Read */
  1757.     if (isset($request_mdn&& $request_mdn{
  1758.         $rfc822_header->dnt $rfc822_header->parseAddress($from_addr,true);
  1759.     elseif (isset($rfc822_header->dnt)) {
  1760.         unset($rfc822_header->dnt);
  1761.     }
  1762.  
  1763.     /* Receipt: On Delivery */
  1764.     if (!empty($request_dr)) {
  1765. //FIXME: it would be better to fiddle with headers inside of the message object or possibly when delivering the message to its destination; is this possible?
  1766.         $rfc822_header->more_headers['Return-Receipt-To'$from_addr;
  1767.     elseif (isset($rfc822_header->more_headers['Return-Receipt-To'])) {
  1768.         unset($rfc822_header->more_headers['Return-Receipt-To']);
  1769.     }
  1770.  
  1771.     /* multipart messages */
  1772.     if (count($composeMessage->entities)) {
  1773.         $message_body new Message();
  1774.         $message_body->body_part $composeMessage->body_part;
  1775.         $composeMessage->body_part '';
  1776.         $mime_header new MessageHeader;
  1777.         $mime_header->type0 'text';
  1778.         $mime_header->type1 'plain';
  1779.         if ($special_encoding{
  1780.             $mime_header->encoding $special_encoding;
  1781.         else {
  1782.             $mime_header->encoding '8bit';
  1783.         }
  1784.         if ($default_charset{
  1785.             $mime_header->parameters['charset'$default_charset;
  1786.         }
  1787.         $message_body->mime_header $mime_header;
  1788.         array_unshift($composeMessage->entities$message_body);
  1789.         $content_type new ContentType('multipart/mixed');
  1790.     else {
  1791.         $content_type new ContentType('text/plain');
  1792.         if ($special_encoding{
  1793.             $rfc822_header->encoding $special_encoding;
  1794.         else {
  1795.             $rfc822_header->encoding '8bit';
  1796.         }
  1797.         if ($default_charset{
  1798.             $content_type->properties['charset']=$default_charset;
  1799.         }
  1800.     }
  1801.  
  1802.     $rfc822_header->content_type $content_type;
  1803.     $composeMessage->rfc822_header $rfc822_header;
  1804.     if ($action == 'reply' || $action == 'reply_all'{
  1805.         global $passed_id$passed_ent_id;
  1806.         $reply_id $passed_id;
  1807.         $reply_ent_id $passed_ent_id;
  1808.     else {
  1809.         $reply_id '';
  1810.         $reply_ent_id '';
  1811.     }
  1812.  
  1813.     /* Here you can modify the message structure just before we hand
  1814.        it over to deliver; plugin authors note that $composeMessage
  1815.        is sent and modified by reference since 1.5.2 */
  1816.     do_hook('compose_send'$composeMessage);
  1817. //TODO: need to migrate to the following, but it neessitates changes in existing plugins, since the args are now an array
  1818.     //$temp = array(&$composeMessage, &$draft);
  1819.     //do_hook('compose_send', $temp);
  1820.  
  1821.     // remove special header if present and prepare to mark
  1822.     // a message that a draft was composed in reply to
  1823.     if (!empty($composeMessage->rfc822_header->x_sm_flag_reply&& !$draft{
  1824.         global $passed_id$mailbox;
  1825.         // tricks the code below that marks the reply
  1826.         list($action$passed_id$mailboxexplode('::'$rfc822_header->x_sm_flag_reply3);
  1827.         unset($composeMessage->rfc822_header->x_sm_flag_reply);
  1828.         unset($composeMessage->rfc822_header->more_headers['X-SM-Flag-Reply']);
  1829.     }
  1830.  
  1831.     if (!$useSendmail && !$draft{
  1832.         require_once(SM_PATH 'class/deliver/Deliver_SMTP.class.php');
  1833.         $deliver new Deliver_SMTP();
  1834.         global $smtpServerAddress$smtpPort$smtp_stream_options$pop_before_smtp$pop_before_smtp_host;
  1835.  
  1836.         $authPop (isset($pop_before_smtp&& $pop_before_smtptrue false;
  1837.         if (empty($pop_before_smtp_host)) $pop_before_smtp_host $smtpServerAddress;
  1838.         get_smtp_user($user$pass);
  1839.         $stream $deliver->initStream($composeMessage,$domain,0,
  1840.                 $smtpServerAddress$smtpPort$user$pass$authPop$pop_before_smtp_host$smtp_stream_options);
  1841.     elseif (!$draft{
  1842.         require_once(SM_PATH 'class/deliver/Deliver_SendMail.class.php');
  1843.         global $sendmail_path$sendmail_args;
  1844.         // Check for outdated configuration
  1845.         if (!isset($sendmail_args)) {
  1846.             if ($sendmail_path=='/var/qmail/bin/qmail-inject'{
  1847.                 $sendmail_args '';
  1848.             else {
  1849.                 $sendmail_args '-i -t';
  1850.             }
  1851.         }
  1852.         $deliver new Deliver_SendMail(array('sendmail_args'=>$sendmail_args));
  1853.         $stream $deliver->initStream($composeMessage,$sendmail_path);
  1854.     elseif ($draft{
  1855.         global $draft_folder;
  1856.         $imap_stream sqimap_login($usernamefalse$imapServerAddress,
  1857.                 $imapPort0$imap_stream_options);
  1858.         if (sqimap_mailbox_exists ($imap_stream$draft_folder)) {
  1859. //TODO: this can leak private information about folders and message IDs if messages are accessed/sent from another client --- should this feature be optional?
  1860.             // make note of the message to mark as having been replied to
  1861.             global $passed_id$mailbox;
  1862.             if ($action == 'reply' || $action == 'reply_all' || $action == 'forward' || $action == 'forward_as_attachment'{
  1863.                 $composeMessage->rfc822_header->more_headers['X-SM-Flag-Reply'$action '::' $passed_id '::' $mailbox;
  1864.             }
  1865.  
  1866.             require_once(SM_PATH 'class/deliver/Deliver_IMAP.class.php');
  1867.             $imap_deliver new Deliver_IMAP();
  1868.             $success $imap_deliver->mail($composeMessage$imap_stream$reply_id$reply_ent_id$imap_stream$draft_folder);
  1869.             sqimap_logout($imap_stream);
  1870.             unset ($imap_deliver);
  1871.             $composeMessage->purgeAttachments();
  1872. //TODO: completely unclear if should be using $compose_session instead of $session below
  1873.             unset($compose_messages[$session]);
  1874.             sqsession_register($compose_messages,'compose_messages');
  1875.             return $success;
  1876.         else {
  1877.             $msg  '<br />'.sprintf(_("Error: Draft folder %s does not exist.")sm_encode_html_special_chars($draft_folder));
  1878.             plain_error_message($msg);
  1879.             return false;
  1880.         }
  1881.     }
  1882.     $success false;
  1883.     if ($stream{
  1884.         $deliver->mail($composeMessage$stream$reply_id$reply_ent_id);
  1885.         $success $deliver->finalizeStream($stream);
  1886.     }
  1887.     if (!$success{
  1888.         // $deliver->dlv_server_msg is not always server's reply
  1889.         $msg _("Message not sent.")
  1890.              . "<br />\n"
  1891.              . (isset($deliver->dlv_msg$deliver->dlv_msg '');
  1892.         if (!empty($deliver->dlv_server_msg)) {
  1893.             // add 'server replied' part only when it is not empty.
  1894.             // Delivery error can be generated by delivery class itself
  1895.             $msg .= '<br />'
  1896.                   . _("Server replied:"' '
  1897.                   . (isset($deliver->dlv_ret_nr$deliver->dlv_ret_nr ' ' '')
  1898.                   . nl2br(sm_encode_html_special_chars($deliver->dlv_server_msg));
  1899.         }
  1900.         plain_error_message($msg);
  1901.     else {
  1902.         unset ($deliver);
  1903.         $imap_stream sqimap_login($usernamefalse$imapServerAddress$imapPort0$imap_stream_options);
  1904.  
  1905.  
  1906.         // mark as replied or forwarded if applicable
  1907.         //
  1908.         global $what$iAccount$startMessage$passed_id$fwduid$mailbox;
  1909.  
  1910.         if ($action=='reply' || $action=='reply_all' || $action=='forward' || $action=='forward_as_attachment'{
  1911.             require(SM_PATH 'functions/mailbox_display.php');
  1912.             // select errors here could be due to a draft reply being sent
  1913.             // after the original message's mailbox is moved or deleted
  1914.             $aMailbox sqm_api_mailbox_select($imap_stream$iAccount$mailbox,array('setindex' => $what'offset' => $startMessage),array()false);
  1915.             // a non-empty return from above means we can proceed
  1916.             if (!empty($aMailbox)) {
  1917.                 switch($action{
  1918.                 case 'reply':
  1919.                 case 'reply_all':
  1920.                     // check if we are allowed to set the \\Answered flag
  1921.                     if (in_array('\\answered',$aMailbox['PERMANENTFLAGS']true)) {
  1922.                         $aUpdatedMsgs sqimap_toggle_flag($imap_streamarray($passed_id)'\\Answered'truefalse);
  1923.                         if (isset($aUpdatedMsgs[$passed_id]['FLAGS'])) {
  1924.                             /**
  1925.                             * Only update the cached headers if the header is
  1926.                             * cached.
  1927.                             */
  1928.                             if (isset($aMailbox['MSG_HEADERS'][$passed_id])) {
  1929.                                 $aMailbox['MSG_HEADERS'][$passed_id]['FLAGS'$aMsg['FLAGS'];
  1930.                             }
  1931.                         }
  1932.                     }
  1933.                     break;
  1934.                 case 'forward':
  1935.                 case 'forward_as_attachment':
  1936.                     // check if we are allowed to set the $Forwarded flag (RFC 4550 paragraph 2.8)
  1937.                     if (in_array('$forwarded',$aMailbox['PERMANENTFLAGS']true||
  1938.                         in_array('\\*',$aMailbox['PERMANENTFLAGS'])) {
  1939.  
  1940.                         // when forwarding as an attachment from the message
  1941.                         // list, passed_id is not used, need to get UID(s)
  1942.                         // from the query string
  1943.                         //
  1944.                         if (empty($passed_id&& !empty($fwduid))
  1945.                             $ids explode('_'$fwduid);
  1946.                         else
  1947.                             $ids array($passed_id);
  1948.  
  1949.                         $aUpdatedMsgs sqimap_toggle_flag($imap_stream$ids'$Forwarded'truefalse);
  1950.  
  1951.                         foreach ($ids as $id{
  1952.                             if (isset($aUpdatedMsgs[$id]['FLAGS'])) {
  1953.                                 if (isset($aMailbox['MSG_HEADERS'][$id])) {
  1954.                                     $aMailbox['MSG_HEADERS'][$id]['FLAGS'$aMsg['FLAGS'];
  1955.                                 }
  1956.                             }
  1957.                         }
  1958.                     }
  1959.                     break;
  1960.                 }
  1961.  
  1962.                 /**
  1963.                  * Write mailbox with updated seen flag information back to cache.
  1964.                  */
  1965.                 if(isset($aUpdatedMsgs[$passed_id])) {
  1966.                     $mailbox_cache[$iAccount.'_'.$aMailbox['NAME']] $aMailbox;
  1967.                     sqsession_register($mailbox_cache,'mailbox_cache');
  1968.                 }
  1969.             }
  1970.         }
  1971.  
  1972.  
  1973.         // move to sent folder
  1974.         //
  1975.         $move_to_sent getPref($data_dir,$username,'move_to_sent');
  1976.         if (isset($default_move_to_sent&& ($default_move_to_sent != 0)) {
  1977.             $svr_allow_sent true;
  1978.         else {
  1979.             $svr_allow_sent false;
  1980.         }
  1981.  
  1982.         if (isset($sent_folder&& (($sent_folder != ''|| ($sent_folder != 'none'))
  1983.                 && sqimap_mailbox_exists$imap_stream$sent_folder)) {
  1984.             $fld_sent true;
  1985.         else {
  1986.             $fld_sent false;
  1987.         }
  1988.  
  1989.         if ((isset($move_to_sent&& ($move_to_sent != 0)) || (!isset($move_to_sent))) {
  1990.             $lcl_allow_sent true;
  1991.         else {
  1992.             $lcl_allow_sent false;
  1993.         }
  1994.  
  1995.         if (($fld_sent && $svr_allow_sent && !$lcl_allow_sent|| ($fld_sent && $lcl_allow_sent)) {
  1996.             if ($action == 'reply' || $action == 'reply_all'{
  1997.                 $save_reply_with_orig=getPref($data_dir,$username,'save_reply_with_orig');
  1998.                 if ($save_reply_with_orig{
  1999.                     $sent_folder $mailbox;
  2000.                 }
  2001.             }
  2002.             require_once(SM_PATH 'class/deliver/Deliver_IMAP.class.php');
  2003.             $imap_deliver new Deliver_IMAP();
  2004.             $imap_deliver->mail($composeMessage$imap_stream$reply_id$reply_ent_id$imap_stream$sent_folder);
  2005.             unset ($imap_deliver);
  2006.         }
  2007.  
  2008.  
  2009.         // final cleanup
  2010.         //
  2011.         $composeMessage->purgeAttachments();
  2012. //TODO: completely unclear if should be using $compose_session instead of $session below
  2013.         unset($compose_messages[$session]);
  2014.         sqsession_register($compose_messages,'compose_messages');
  2015.         sqimap_logout($imap_stream);
  2016.  
  2017.     }
  2018.     return $success;
  2019. }

Documentation generated on Mon, 13 Jan 2020 04:22:09 +0100 by phpDocumentor 1.4.3