chagenbu Wed Feb 7 12:22:59 2001 EDT Added files: /php4/pear/Mail RFC822.php SMTP.php Sendmail.php Removed files: /php4/pear/Mail rfc822.php sendmail.php smtp.php Modified files: /php4/pear Mail.php Makefile.in Log: may as well go by the capitalization we agreed on sooner rather than later.
Index: php4/pear/Mail.php diff -u php4/pear/Mail.php:1.8 php4/pear/Mail.php:1.9 --- php4/pear/Mail.php:1.8 Tue Jan 9 17:01:52 2001 +++ php4/pear/Mail.php Wed Feb 7 12:22:59 2001 @@ -33,16 +33,20 @@ * @param array The parameters to pass to the Mail:: object. * @access public */ - function factory($mailer_type, $params = array()) + function factory($driver, $params = array()) { - $mailer_type = strtolower($mailer_type); - $classfile = PEAR_INSTALL_DIR . '/Mail/' . $mailer_type . '.php'; - if (@is_readable($classfile)) { - include_once $classfile; - $class = 'Mail_' . $mailer_type; + if (@include_once 'Mail/' . $driver . '.php') { + $class = 'Mail_' . $driver; + } elseif (@include_once 'Mail/' . strtoupper($driver) . '.php') { + $class = 'Mail_' . strtoupper($driver); + } elseif (@include_once 'Mail/' . ucfirst($driver) . '.php') { + $class = 'Mail_' . ucfirst($driver); + } + + if (isset($class)) { return new $class($params); } else { - return new PEAR_Error('unable to find classfile: ' . $classfile); + return new PEAR_Error('Unable to find class for driver ' . $driver); } } @@ -116,9 +120,9 @@ foreach ($headers as $key => $val) { if ($key == 'From') { - include_once 'Mail/rfc822.php'; + include_once 'Mail/RFC822.php'; - $from_arr = Mail_rfc822::parseAddressList($val, 'localhost', false); + $from_arr = Mail_RFC822::parseAddressList($val, 'localhost', false); $from = $from_arr[0]->mailbox . '@' . $from_arr[0]->host; if (strstr($from, ' ')) { // Reject outright envelope From addresses with spaces. @@ -152,7 +156,7 @@ */ function parseRecipients($recipients) { - include_once 'Mail/rfc822.php'; + include_once 'Mail/RFC822.php'; // if we're passed an array, assume addresses are valid and // implode them before parsing. @@ -163,7 +167,7 @@ // Parse recipients, leaving out all personal info. This is // for smtp recipients, etc. All relevant personal information // should already be in the headers. - $addresses = Mail_rfc822::parseAddressList($recipients, 'localhost', false); + $addresses = Mail_RFC822::parseAddressList($recipients, 'localhost', false); $recipients = array(); if (is_array($addresses)) { foreach ($addresses as $ob) { Index: php4/pear/Makefile.in diff -u php4/pear/Makefile.in:1.66 php4/pear/Makefile.in:1.67 --- php4/pear/Makefile.in:1.66 Tue Jan 23 13:09:55 2001 +++ php4/pear/Makefile.in Wed Feb 7 12:22:59 2001 @@ -68,9 +68,9 @@ Log/sql.php \ Log/syslog.php \ Mail.php \ - Mail/rfc822.php \ - Mail/sendmail.php \ - Mail/smtp.php \ + Mail/RFC822.php \ + Mail/Sendmail.php \ + Mail/SMTP.php \ Math/Fraction.php \ Math/Util.php \ Net/Curl.php \ Index: php4/pear/Mail/RFC822.php +++ php4/pear/Mail/RFC822.php <?php // // +----------------------------------------------------------------------+ // | PHP version 4.0 | // +----------------------------------------------------------------------+ // | Copyright (c) 1997-2001 The PHP Group | // +----------------------------------------------------------------------+ // | This source file is subject to version 2.02 of the PHP license, | // | that is bundled with this package in the file LICENSE, and is | // | available at through the world-wide-web at | // | http://www.php.net/license/2_02.txt. | // | If you did not receive a copy of the PHP license and are unable to | // | obtain it through the world-wide-web, please send a note to | // | [EMAIL PROTECTED] so we can mail you a copy immediately. | // +----------------------------------------------------------------------+ // | Authors: Richard Heyes <[EMAIL PROTECTED]> | // | Chuck Hagenbuch <[EMAIL PROTECTED]> | // +----------------------------------------------------------------------+ /** * RFC 822 Email address list validation Utility * * @author Richard Heyes <[EMAIL PROTECTED]> * @author Chuck Hagenbuch <[EMAIL PROTECTED]> * @version $Revision: 1.1 $ */ class Mail_RFC822 { // Properties (All private) var $address = null; var $default_host = 'localhost'; var $nestGroups = true; var $addresses = array(); var $structure = array(); var $error = null; var $index = null; var $num_groups = 0; var $mailRFC822 = true; /** * Sets up the object. The address must either be set here or when * calling parseAddressList(). One or the other. * * @access public * @param $address The address(es) to validate. * @param $default_host (Optional) Default domain/host etc. If not supplied, will be set to localhost. * @param $nest_groups (Optional) Whether to return the structure with groups nested for easier viewing. */ function Mail_RFC822($address = null, $default_host = null, $nest_groups = null) { if (isset($address)) $this->address = $address; if (isset($default_host)) $this->default_host = $default_host; if (isset($nest_groups)) $this->nestGroups = $nest_groups; } /** * Starts the whole process. The address must either be set here * or when creating the object. One or the other. * * @access public * @param $address The address(es) to validate. * @param $default_host (Optional) Default domain/host etc. * @param $nest_groups (Optional) Whether to return the structure with groups nested for easier viewing. */ function parseAddressList($address = null, $default_host = null, $nest_groups = null) { if (!isset($this->mailRFC822)) { $obj = new Mail_RFC822($address, $default_host, $nest_groups); return $obj->parseAddressList(); } if (isset($address)) $this->address = $address; if (isset($default_host)) $this->default_host = $default_host; if (isset($nest_groups)) $this->nestGroups = $nest_groups; $this->structure = array(); $this->addresses = array(); $this->error = null; $this->index = null; if (!$this->splitAddresses($this->address) || isset($this->error)) { return (array)$this->error; } for ($i = 0; $i < count($this->addresses); $i++){ if (($return = $this->validateAddress($this->addresses[$i])) === false || isset($this->error)) { return (array)$this->error; } if (!$this->nestGroups) { $this->structure = array_merge($this->structure, $return); } else { $this->structure[] = $return; } } return $this->structure; } /** * Splits an address into seperate addresses. * * @access private * @param $address The addresses to split. */ function splitAddresses($address = '') { if ($this->isGroup($address) && !isset($this->error)) { $split_char = ';'; $is_group = true; } elseif (!isset($this->error)) { $split_char = ','; $is_group = false; } elseif (isset($this->error)) { return false; } // Split the string based on the above ten or so lines. $parts = explode($split_char, $address); $string = $this->splitCheck($parts, $split_char); // If a group... if ($is_group) { // If $string does not contain a colon outside of // brackets/quotes etc then something's fubar. // First check there's a colon at all: if (strpos($string, ':') === false) { $this->error = 'Invalid address: ' . $string; return false; } // Now check it's outside of brackets/quotes: if (!$this->splitCheck(explode(':', $string), ':')) return false; // We must have a group at this point, so increase the counter: $this->num_groups++; } // $string now contains the first full address/group. // Add to the addresses array. $this->addresses[] = array( 'address' => trim($string), 'group' => $is_group ); // Remove the now stored address from the initial line, the +1 // is to account for the explode character. $address = trim(substr($address, strlen($string) + 1)); // If the next char is a comma and this was a group, then // there are more addresses, otherwise, if there are any more // chars, then there is another address. if ($is_group && substr($address, 0, 1) == ','){ $address = trim(substr($address, 1)); $this->splitAddresses($address); return true; } elseif (strlen($address) > 0) { $this->splitAddresses($address); return true; } else { return true; } // If you got here then something's off return false; } /** * Checks for a group at the start of the string. * * @access private * @param $address The address to check. */ function isGroup($address) { // First comma not in quotes, angles or escaped: $parts = explode(',', $address); $string = $this->splitCheck($parts, ','); // Now we have the first address, we can reliably check for a // group by searching for a colon that's not escaped or in // quotes or angle brackets. if (count($parts = explode(':', $string)) > 1) { $string2 = $this->splitCheck($parts, ':'); return ($string2 !== $string); } else { return false; } } /** * A common function that will check an exploded string. * * @access private * @param $parts The exloded string. * @param $char The char that was exploded on. */ function splitCheck($parts, $char) { $string = $parts[0]; for ($i = 0; $i < count($parts); $i++) { if ($this->hasUnclosedQuotes($string) || $this->hasUnclosedBrackets($string, '<>') || $this->hasUnclosedBrackets($string, '[]') || substr($string, -1) == '\\') { if (isset($parts[$i + 1])) { $string = $string . $char . $parts[$i + 1]; } else { $this->error = 'Invalid address spec. Unclosed bracket or quotes'; return false; } } else { $this->index = $i; break; } } return $string; } /** * Checks if a string has an unclosed quotes or not. * * @access private * @param $string The string to check. */ function hasUnclosedQuotes($string) { $string = explode('"', $string); $string_cnt = count($string); for ($i = 0; $i < (count($string) - 1); $i++) if (substr($string[$i], -1) == '\\') $string_cnt--; return ($string_cnt % 2 === 0); } /** * Checks if a string has an unclosed brackets or not. IMPORTANT: * This function handles both angle brackets and square brackets; * * @access private * @param $string The string to check. * @param $chars The characters to check for. */ function hasUnclosedBrackets($string, $chars) { $num_angle_start = substr_count($string, $chars[0]); $num_angle_end = substr_count($string, $chars[1]); $this->hasUnclosedBracketsSub($string, $num_angle_start, $chars[0]); $this->hasUnclosedBracketsSub($string, $num_angle_end, $chars[1]); if ($num_angle_start < $num_angle_end) { $this->error = 'Invalid address spec. Unmatched quote or bracket (' . $chars . ')'; return false; } else { return ($num_angle_start > $num_angle_end); } } /** * Sub function that is used only by hasUnclosedBrackets(). * * @access private * @param $string The string to check. * @param $num The number of occurences. * @param $char The character to count. */ function hasUnclosedBracketsSub($string, &$num, $char) { $parts = explode($char, $string); for ($i = 0; $i < count($parts); $i++){ if (substr($parts[$i], -1) == '\\' || $this->hasUnclosedQuotes($parts[$i])) $num--; if (isset($parts[$i + 1])) $parts[$i + 1] = $parts[$i] . $char . $parts[$i + 1]; } return $num; } /** * Function to begin checking the address. * * @access private * @param $address The address to validate. */ function validateAddress($address) { $is_group = false; if ($address['group']) { $is_group = true; // Get the group part of the name $parts = explode(':', $address['address']); $groupname = $this->splitCheck($parts, ':'); $structure = array(); // And validate the group part of the name. if (!$this->validatePhrase($groupname)){ $this->error = 'Group name did not validate.'; return false; } else { // Don't include groups if we are not nesting // them. This avoids returning invalid addresses. if ($this->nestGroups) { $structure = new stdClass; $structure->groupname = $groupname; } } $address['address'] = ltrim(substr($address['address'], strlen($groupname . ':'))); } // If a group then split on comma and put into an array. // Otherwise, Just put the whole address in an array. if ($is_group) { while (strlen($address['address']) > 0) { $parts = explode(',', $address['address']); $addresses[] = $this->splitCheck($parts, ','); $address['address'] = trim(substr($address['address'], strlen(end($addresses) . ','))); } } else { $addresses[] = $address['address']; } // Check that $addresses is set, if address like this: // Groupname:; // Then errors were appearing. if (!isset($addresses)){ $this->error[] = 'Empty group.'; return false; } for ($i = 0; $i < count($addresses); $i++) { $addresses[$i] = trim($addresses[$i]); } // Validate each mailbox. // Format could be one of: name <[EMAIL PROTECTED]> // [EMAIL PROTECTED] // geezer // ... or any other format valid by RFC 822. array_walk($addresses, array($this, 'validateMailbox')); // Nested format if ($this->nestGroups) { if ($is_group) { $structure->addresses = $addresses; } else { $structure = $addresses[0]; } // Flat format } else { if ($is_group) { $structure = array_merge($structure, $addresses); } else { $structure = $addresses; } } return $structure; } /** * Function to validate a phrase. * * @access private * @param $phrase The phrase to check. */ function validatePhrase($phrase) { // Splits on one or more Tab or space. $parts = preg_split('/[ \\x09]+/', $phrase, 0, PREG_SPLIT_NO_EMPTY); $phrase_parts = array(); while (count($parts) > 0){ $phrase_parts[] = $this->splitCheck($parts, ' '); for ($i = 0; $i < $this->index + 1; $i++) array_shift($parts); } for ($i = 0; $i < count($phrase_parts); $i++) { // If quoted string: if (substr($phrase_parts[$i], 0, 1) == '"') { if (!$this->validateQuotedString($phrase_parts[$i])) return false; continue; } // Otherwise it's an atom: if (!$this->validateAtom($phrase_parts[$i])) return false; } return true; } /** * Function to validate an atom which from rfc822 is: * atom = 1*<any CHAR except specials, SPACE and CTLs> * * @access private * @param $atom The string to check. */ function validateAtom($atom) { // Check for any char from ASCII 0 - ASCII 127 if (!preg_match('/^[\\x00-\\x7E]+$/i', $atom, $matches)) return false; // Check for specials: if (preg_match('/[][()<>@,;\\:". ]/', $atom)) return false; // Check for control characters (ASCII 0-31): if (preg_match('/[\\x00-\\x1F]+/', $atom)) return false; return true; } /** * Function to validate quoted string, which is: * quoted-string = <"> *(qtext/quoted-pair) <"> * * @access private * @param $qstring The string to check. */ function validateQuotedString($qstring) { // Leading and trailing " $qstring = substr($qstring, 1, -1); // Perform check. return !(preg_match('/(.)[\x0D\\\\"]/', $qstring, $matches) && $matches[1] != '\\'); } /** * Function to validate a mailbox, which is: * mailbox = addr-spec ; simple address * / phrase route-addr ; name and route-addr * * @access private * @param $mailbox The string to check. */ function validateMailbox(&$mailbox) { // A couple of defaults. $phrase = ''; // Check for name + route-addr if (substr($mailbox, -1) == '>' && substr($mailbox, 0, 1) != '<') { $parts = explode('<', $mailbox); $name = $this->splitCheck($parts, '<'); $phrase = trim($name); $route_addr = trim(substr($mailbox, strlen($name.'<'), -1)); if ($this->validatePhrase($phrase) === false || ($route_addr = $this->validateRouteAddr($route_addr)) === false) return false; // Only got addr-spec } else { // First snip angle brackets if present. if (substr($mailbox,0,1) == '<' && substr($mailbox,-1) == '>') $addr_spec = substr($mailbox,1,-1); else $addr_spec = $mailbox; if (($addr_spec = $this->validateAddrSpec($addr_spec)) === false) return false; } // Construct the object that will be returned. $mbox = new stdClass(); $phrase !== '' ? $mbox->personal = $phrase : ''; if (isset($route_addr)) { $mbox->mailbox = $route_addr['local_part']; $mbox->host = $route_addr['domain']; $route_addr['adl'] !== '' ? $mbox->adl = $route_addr['adl'] : ''; } else { $mbox->mailbox = $addr_spec['local_part']; $mbox->host = $addr_spec['domain']; } $mailbox = $mbox; } /** * This function validates a route-addr which is: * route-addr = "<" [route] addr-spec ">" * * Angle brackets have already been removed at the point of * getting to this function. * * @access private * @param $route_addr The string to check. */ function validateRouteAddr($route_addr) { // Check for colon. if (strpos($route_addr, ':') !== false) { $parts = explode(':', $route_addr); $route = $this->splitCheck($parts, ':'); } else { $route = $route_addr; } // If $route is same as $route_addr then the colon was in // quotes or brackets or, of course, non existent. if ($route === $route_addr){ unset($route); $addr_spec = $route_addr; if (($addr_spec = $this->validateAddrSpec($addr_spec)) === false) { return false; } } else { // Validate route part. if (($route = $this->validateRoute($route)) === false) { return false; } $addr_spec = substr($route_addr, strlen($route . ':')); // Validate addr-spec part. if (($addr_spec = $this->validateAddrSpec($addr_spec)) === false) { return false; } } if (isset($route)) { $return['adl'] = $route; } else { $return['adl'] = ''; } $return = array_merge($return, $addr_spec); return $return; } /** * Function to validate a route, which is: * route = 1#("@" domain) ":" * * @access private * @param $route The string to check. */ function validateRoute($route) { // Split on comma. $domains = explode(',', trim($route)); for ($i = 0; $i < count($domains); $i++) { $domains[$i] = str_replace('@', '', trim($domains[$i])); if (!$this->validateDomain($domains[$i])) return false; } return $route; } /** * Function to validate a domain, though this is not quite what * you expect of a strict internet domain. * * domain = sub-domain *("." sub-domain) * * @access private * @param $domain The string to check. */ function validateDomain($domain) { // Note the different use of $subdomains and $sub_domains $subdomains = explode('.', $domain); while (count($subdomains) > 0) { $sub_domains[] = $this->splitCheck($subdomains, '.'); for ($i = 0; $i < $this->index + 1; $i++) array_shift($subdomains); } for ($i = 0; $i < count($sub_domains); $i++) { if (!$this->validateSubdomain(trim($sub_domains[$i]))) return false; } // Managed to get here, so return input. return $domain; } /** * Function to validate a subdomain: * subdomain = domain-ref / domain-literal * * @access private * @param $subdomain The string to check. */ function validateSubdomain($subdomain) { if (preg_match('|^\[(.*)]$|', $subdomain, $arr)){ if (!$this->validateDliteral($arr[1])) return false; } else { if (!$this->validateAtom($subdomain)) return false; } // Got here, so return successful. return true; } /** * Function to validate a domain literal: * domain-literal = "[" *(dtext / quoted-pair) "]" * * @access private * @param $dliteral The string to check. */ function validateDliteral($dliteral) { return !preg_match('/(.)[][\x0D\\\\]/', $dliteral, $matches) && $matches[1] != '\\'; } /** * Function to validate an addr-spec. * * addr-spec = local-part "@" domain * * @access private * @param $addr_spec The string to check. */ function validateAddrSpec($addr_spec) { $addr_spec = trim($addr_spec); // Split on @ sign if there is one. if (strpos($addr_spec, '@') !== false) { $parts = explode('@', $addr_spec); $local_part = $this->splitCheck($parts, '@'); $domain = substr($addr_spec, strlen($local_part . '@')); // No @ sign so assume the default domain. } else { $local_part = $addr_spec; $domain = $this->default_host; } if (($local_part = $this->validateLocalPart($local_part)) === false) return false; if (($domain = $this->validateDomain($domain)) === false) return false; // Got here so return successful. return array('local_part' => $local_part, 'domain' => $domain); } /** * Function to validate the local part of an address: * local-part = word *("." word) * * @access private * @param $local_part */ function validateLocalPart($local_part) { $parts = explode('.', $local_part); // Split the local_part into words. while (count($parts) > 0){ $words[] = $this->splitCheck($parts, '.'); for ($i = 0; $i < $this->index + 1; $i++) { array_shift($parts); } } // Validate each word. for ($i = 0; $i < count($words); $i++) { if ($this->validatePhrase(trim($words[$i])) === false) return false; } // Managed to get here, so return the input. return $local_part; } } ?> Index: php4/pear/Mail/SMTP.php +++ php4/pear/Mail/SMTP.php <?php // // +----------------------------------------------------------------------+ // | PHP version 4.0 | // +----------------------------------------------------------------------+ // | Copyright (c) 1997-2001 The PHP Group | // +----------------------------------------------------------------------+ // | This source file is subject to version 2.02 of the PHP license, | // | that is bundled with this package in the file LICENSE, and is | // | available at through the world-wide-web at | // | http://www.php.net/license/2_02.txt. | // | If you did not receive a copy of the PHP license and are unable to | // | obtain it through the world-wide-web, please send a note to | // | [EMAIL PROTECTED] so we can mail you a copy immediately. | // +----------------------------------------------------------------------+ // | Authors: Chuck Hagenbuch <[EMAIL PROTECTED]> | // +----------------------------------------------------------------------+ require_once 'Mail.php'; /** * SMTP implementation of the PEAR Mail:: interface. Requires the PEAR * Net_SMTP:: class. */ class Mail_SMTP extends Mail { /** * The SMTP host to connect to. * @var string */ var $host = 'localhost'; /** * The port the SMTP server is on. * @var int */ var $port = 25; /** * Whether or not to attempt to authenticate to the SMTP server. * @var boolean */ var $auth = false; /** * The username to use if the SMTP server requires authentication. * @var string */ var $username = ''; /** * The password to use if the SMTP server requires authentication. * @var string */ var $password = ''; /** * Constructor. * * Instantiates a new Mail_SMTP:: object based on the parameters * passed in. It looks for the following parameters: * host The server to connect to. Defaults to localhost. * port The port to connect to. Defaults to 25. * auth Whether or not to use SMTP auth. Defaults to false. * username The username to use for SMTP auth. No default. * password The password to use for SMTP auth. No default. * * If a parameter is present in the $params array, it replaces the * default. * * @param array Hash containing any parameters different from the * defaults. */ function Mail_SMTP($params) { if (isset($params['host'])) $this->host = $params['host']; if (isset($params['port'])) $this->port = $params['port']; if (isset($params['auth'])) $this->auth = $params['auth']; if (isset($params['username'])) $this->username = $params['username']; if (isset($params['password'])) $this->password = $params['password']; } /** * Implements Mail::send() function using SMTP. * * @param mixed Either a comma-seperated list of recipients * (RFC822 compliant), or an array of recipients, * each RFC822 valid. This may contain recipients not * specified in the headers, for Bcc:, resending * messages, etc. * * @param array The array of headers to send with the mail, in an * associative array, where the array key is the * header name (ie, 'Subject'), and the array value * is the header value (ie, 'test'). The header * produced from those values would be 'Subject: * test'. * * @param string The full text of the message body, including any * Mime parts, etc. * * @return mixed Returns true on success, or a PEAR_Error * containing a descriptive error message on * failure. * @access public */ function send($recipients, $headers, $body) { include_once 'Net/SMTP.php'; if (!($smtp = new Net_SMTP($this->host, $this->port))) { return new PEAR_Error('unable to instantiate Net_SMTP object'); } if (!$smtp->connect()) { return new PEAR_Error('unable to connect to smtp server ' . $this->host . ':' . $this->port); } if ($this->auth) { if (!$smtp->auth($this->username, $this->password)) { return new PEAR_Error('unable to authenticate to smtp server'); } if (!$smtp->identifySender()) { return new PEAR_Error('unable to identify smtp server'); } } list($from, $text_headers) = $this->prepareHeaders($headers); if (!isset($from)) { return new PEAR_Error('No from address given'); } if (!($smtp->mailFrom($from))) { return new PEAR_Error('unable to set sender to [' . $from . ']'); } $recipients = $this->parseRecipients($recipients); foreach($recipients as $recipient) { if (!$smtp->rcptTo($recipient)) { return new PEAR_Error('unable to add recipient [' . $recipient . ']'); } } if (!$smtp->data($text_headers . "\n" . $body)) { return new PEAR_Error('unable to send data'); } $smtp->disconnect(); return true; } } ?> Index: php4/pear/Mail/Sendmail.php +++ php4/pear/Mail/Sendmail.php <?php // // +----------------------------------------------------------------------+ // | PHP version 4.0 | // +----------------------------------------------------------------------+ // | Copyright (c) 1997-2001 The PHP Group | // +----------------------------------------------------------------------+ // | This source file is subject to version 2.02 of the PHP license, | // | that is bundled with this package in the file LICENSE, and is | // | available at through the world-wide-web at | // | http://www.php.net/license/2_02.txt. | // | If you did not receive a copy of the PHP license and are unable to | // | obtain it through the world-wide-web, please send a note to | // | [EMAIL PROTECTED] so we can mail you a copy immediately. | // +----------------------------------------------------------------------+ // | Authors: Chuck Hagenbuch <[EMAIL PROTECTED]> | // +----------------------------------------------------------------------+ require_once 'Mail.php'; /** * Sendmail implementation of the PEAR Mail:: interface. */ class Mail_Sendmail extends Mail { /** * The location of the sendmail binary on the filesystem. * @var string */ var $sendmail_path = '/usr/sbin/sendmail'; /** * Constructor. * * Instantiates a new Mail_Sendmail:: object based on the parameters * passed in. It looks for the following parameters: * sendmail_path The location of the sendmail binary on the * filesystem. Defaults to '/usr/sbin/sendmail'. * * If a parameter is present in the $params array, it replaces the * default. * * @param array Hash containing any parameters different from the * defaults. */ function Mail_Sendmail($params) { if (isset($params['sendmail_path'])) $this->sendmail_path = $params['sendmail_path']; } /** * Implements Mail::send() function using the sendmail * command-line binary. * * @param mixed Either a comma-seperated list of recipients * (RFC822 compliant), or an array of recipients, * each RFC822 valid. This may contain recipients not * specified in the headers, for Bcc:, resending * messages, etc. * * @param array The array of headers to send with the mail, in an * associative array, where the array key is the * header name (ie, 'Subject'), and the array value * is the header value (ie, 'test'). The header * produced from those values would be 'Subject: * test'. * * @param string The full text of the message body, including any * Mime parts, etc. * * @return mixed Returns true on success, or a PEAR_Error * containing a descriptive error message on * failure. * @access public */ function send($recipients, $headers, $body) { $recipients = escapeShellCmd(implode(' ', $this->parseRecipients($recipients))); list($from, $text_headers) = $this->prepareHeaders($headers); if (!isset($from)) { return new PEAR_Error('No from address given'); } $result = 0; if (@is_executable($this->sendmail_path)) { $from = '"' . escapeShellCmd($from) . '"'; $mail = popen($this->sendmail_path . " -i -f$from -- $recipients", 'w'); fputs($mail, $text_headers); fputs($mail, "\n"); // newline to end the headers section fputs($mail, $body); $result = pclose($mail) >> 8 & 0xFF; // need to shift the pclose result to get the exit code } else { return new PEAR_Error('sendmail [' . $this->sendmail_path . '] not executable'); } // Return. if ($result != 0) { return new PEAR_Error('sendmail returned error code ' . $result); } return true; } } ?>
-- PHP CVS Mailing List (http://www.php.net/) To unsubscribe, e-mail: [EMAIL PROTECTED] For additional commands, e-mail: [EMAIL PROTECTED] To contact the list administrators, e-mail: [EMAIL PROTECTED]