Skip to content

Commit

Permalink
Add info & test case for CRAM-MD5 auth
Browse files Browse the repository at this point in the history
Fix validation test
Fix misdocumented do_debug property in SMTP
Code cleanup
  • Loading branch information
Synchro committed Dec 4, 2012
1 parent c7ff6fb commit 95cc8b8
Show file tree
Hide file tree
Showing 4 changed files with 66 additions and 43 deletions.
4 changes: 2 additions & 2 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@
- Multipart/alternative emails for mail clients that do not read HTML email
- Support for 8bit, base64, binary, and quoted-printable encoding
- Uses the same methods as the very popular AspEmail active server (COM) component
- SMTP authentication
- SMTP authentication with LOGIN, PLAIN, NTLM and CRAM-MD5 mechanisms
- Native language support
- Word wrap
- Compatible with PHP 5.0 and later
Expand All @@ -26,7 +26,7 @@ The PHP mail() function usually sends via a local mail server, typically fronted

## License

This software is licenced under the [LGPL](http://www.gnu.org/licenses/lgpl-2.1.html). Please read LICENSE for information on the
This software is licenced under the [LGPL 2.1](http://www.gnu.org/licenses/lgpl-2.1.html). Please read LICENSE for information on the
software availability and distribution.

## Installation
Expand Down
6 changes: 3 additions & 3 deletions class.phpmailer.php
Original file line number Diff line number Diff line change
Expand Up @@ -260,7 +260,7 @@ class PHPMailer {
public $Password = '';

/**
* Sets SMTP auth type. Options are LOGIN | PLAIN | NTLM (default LOGIN)
* Sets SMTP auth type. Options are LOGIN | PLAIN | NTLM | CRAM-MD5 (default LOGIN)
* @var string
*/
public $AuthType = '';
Expand Down Expand Up @@ -2069,8 +2069,8 @@ public function EncodeQ($str, $position = 'text') {

case 'comment':
$pattern = '\(\)"';
//note that we dont break here!
//for this reason we build the $pattern withoud including delimiters and []
//note that we don't break here!
//for this reason we build the $pattern without including delimiters and []

case 'text':
default:
Expand Down
60 changes: 31 additions & 29 deletions class.smtp.php
Original file line number Diff line number Diff line change
Expand Up @@ -58,10 +58,10 @@ class SMTP {
public $CRLF = "\r\n";

/**
* Sets whether debugging is turned on
* @var bool
* Sets debug output level; 0 for no output
* @var int
*/
public $do_debug; // the level of debug to perform
public $do_debug = 0;

/**
* Sets the function/method to use for debugging output.
Expand Down Expand Up @@ -432,11 +432,11 @@ public function Authenticate($username, $password, $authtype='LOGIN', $realm='',
return false;
}

// Get the challenge
$challenge = base64_decode(substr($rply,4));
// Get the challenge
$challenge = base64_decode(substr($rply,4));

// Build the response
$response = $username . ' ' . $this->hmac($challenge, $password);
// Build the response
$response = $username . ' ' . $this->hmac($challenge, $password);

// Send encoded credentials
fputs($this->smtp_conn, base64_encode($response) . $this->CRLF);
Expand All @@ -462,31 +462,33 @@ public function Authenticate($username, $password, $authtype='LOGIN', $realm='',
/**
* Works like hash_hmac('md5', $data, $key) in case that function is not available
* @access private
* @param string $data
* @param string $key
* @return string
*/
private function hmac($data, $key) {
if (function_exists('hash_hmac')) {
return hash_hmac('md5', $data, $key);
}

// The following borrowed from http://php.net/manual/en/function.mhash.php#27225

// RFC 2104 HMAC implementation for php.
// Creates an md5 HMAC.
// Eliminates the need to install mhash to compute a HMAC
// Hacked by Lance Rushing

$b = 64; // byte length for md5
if (strlen($key) > $b) {
$key = pack("H*",md5($key));
}
$key = str_pad($key, $b, chr(0x00));
$ipad = str_pad('', $b, chr(0x36));
$opad = str_pad('', $b, chr(0x5c));
$k_ipad = $key ^ $ipad ;
$k_opad = $key ^ $opad;

return md5($k_opad . pack("H*",md5($k_ipad . $data)));
if (function_exists('hash_hmac')) {
return hash_hmac('md5', $data, $key);
}

// The following borrowed from http://php.net/manual/en/function.mhash.php#27225

// RFC 2104 HMAC implementation for php.
// Creates an md5 HMAC.
// Eliminates the need to install mhash to compute a HMAC
// Hacked by Lance Rushing

$b = 64; // byte length for md5
if (strlen($key) > $b) {
$key = pack("H*",md5($key));
}
$key = str_pad($key, $b, chr(0x00));
$ipad = str_pad('', $b, chr(0x36));
$opad = str_pad('', $b, chr(0x5c));
$k_ipad = $key ^ $ipad ;
$k_opad = $key ^ $opad;

return md5($k_opad . pack("H*",md5($k_ipad . $data)));
}

/**
Expand Down
39 changes: 30 additions & 9 deletions test/phpmailerTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,6 @@
* @author Andy Prevost
* @author Marcus Bointon
* @copyright 2004 - 2009 Andy Prevost
* @version $Id: phpmailerTest.php 444 2009-05-05 11:22:26Z coolbru $
* @license http://www.gnu.org/copyleft/lesser.html GNU Lesser General Public License
*/

Expand Down Expand Up @@ -285,6 +284,28 @@ function SetAddress($sAddress, $sName = '', $sType = 'to')
// UNIT TESTS
/////////////////////////////////////////////////

/**
* Test CRAM-MD5 authentication
* Needs a connection to a server that supports this auth mechanism, so commented out by default
*/
function testAuthCRAMMD5()
{
$this->Mail->Host = 'hostname';
$this->Mail->Port = 587;
$this->Mail->SMTPAuth = true;
$this->Mail->SMTPSecure = 'tls';
$this->Mail->AuthType = 'CRAM-MD5';
$this->Mail->Username = 'username';
$this->Mail->Password = 'password';
$this->Mail->Body = 'Test body';
$this->Mail->Subject .= ": Auth CRAM-MD5";
$this->Mail->From = '[email protected]';
$this->Mail->Sender = '[email protected]';
$this->Mail->ClearAllRecipients();
$this->Mail->AddAddress('[email protected]');
//$this->assertTrue($this->Mail->Send(), $this->Mail->ErrorInfo);
}

/**
* Test email address validation
* Test addresses obtained from http://isemail.info
Expand Down Expand Up @@ -575,14 +596,14 @@ function testValidate()
}
$err = '';
if (count($goodfails) > 0) {
$err = "Good addreses that failed validation:\n";
$err .= "Good addreses that failed validation:\n";
$err .= implode("\n", $goodfails);
}
if (count($badpasses) > 0) {
if (!empty($err)) {
$err .="\n\n";
}
$err = "Bad addreses that passed validation:\n";
$err .= "Bad addreses that passed validation:\n";
$err .= implode("\n", $badpasses);
}
$this->assertEmpty($err, $err);
Expand Down Expand Up @@ -951,7 +972,7 @@ function test_Translations()
{
$this->Mail->SetLanguage('en');
$definedStrings = $this->Mail->GetTranslations();
$err = '';
$err = '';
foreach (new DirectoryIterator('../language') as $fileInfo) {
if ($fileInfo->isDot()) {
continue;
Expand All @@ -965,14 +986,14 @@ function test_Translations()
$missing = array_diff(array_keys($definedStrings), array_keys($PHPMAILER_LANG));
$extra = array_diff(array_keys($PHPMAILER_LANG), array_keys($definedStrings));
if (!empty($missing)) {
$err .= "Missing translations in $lang: " . implode(', ', $missing)."\n";
$err .= "Missing translations in $lang: " . implode(', ', $missing)."\n";
}
if (!empty($extra)) {
$err .= "Extra translations in $lang: " . implode(', ', $extra)."\n";
}
if (!empty($extra)) {
$err .= "Extra translations in $lang: " . implode(', ', $extra)."\n";
}
}
}
$this->assertEmpty($err, $err);
$this->assertEmpty($err, $err);
}

/**
Expand Down

0 comments on commit 95cc8b8

Please sign in to comment.