Я работал над классом в PHP для отправки почты, и я решил использовать Zend Framework. Этот класс отправляет почту с использованием конфигурации SMTP пользователя. На данный момент я проверяю конфигурацию SMTP пользователя, используя предоставленные учетные данные пользователя, и отправляю «фиктивное» электронное письмо на «фиктивный» адрес электронной почты, и ловлю ZendException
класс, который может быть брошен по ошибке. Это ужасный метод по многим причинам:
Ниже приведен пример того, что я делаю сейчас, чтобы проверить правильность конфигурации SMTP:
public function validSMTP () {
// Get the user's SMTP configuration
$config = $this->getConfiguration ();
// Create a new Zend transport SMTP object
$transport = new Zend_Mail_Transport_Smtp ( $config ["hostname"], [
"auth" => "login",
"ssl" => $config ["protocol"],
"port" => $config ["port"],
"username" => $config ["from"],
"password" => $config ["password"]
]);
// Create a new message and send it to dummy email
$mail = new Zend_Mail ("UTF-8");
$mail->setBodyText ( "null" );
$mail->setFrom ( $config ["from"] );
$mail->addTo ( "[email protected]" );
$mail->setSubject ( "Test" );
// Attempt to send the email
try {
// Send the email out
$mail->send ( $transport );
// If all is well, return true
return true;
}
// Catch all Zend exceptions
catch ( Zend_Exception $exception ) {
// Invalid configuration
return false;
}
}
Итак, мой вопрос: есть ли лучший способ сделать это? Есть ли Zend_Mail есть встроенная функция, как это? Я просмотрел все и не смог найти ничего встроенного в Zend_Mail. Заранее спасибо всем, кто ответит!
Я не вижу ничего в документация это говорит о том, что есть способ просто проверить SMTP-аутентификацию, фактически не пытаясь отправить почту. Может быть, когда вы звоните setOptions
или же setConnectionConfig
он попытается войти в систему и выдаст исключение в случае сбоя, но я подозреваю, что нет.
Вместо того, чтобы выполнять проверку каждый раз, когда вы хотите отправить почту, сохраните данные конфигурации и результаты проверки в файле. Затем, прежде чем выполнять проверку SMTP, прочитайте файл и проверьте, совпадают ли настройки конфигурации. Если они есть, просто верните сохраненные результаты.
Это решает первые два элемента в вашем списке проблем, и они получат отказавшее электронное письмо только в первый раз, если их конфигурация неверна.
Я решил проверить вручную, я опубликую конечный результат ниже, чтобы он мог помочь кому-то в будущем. Спасибо @Barmar за помощь в поиске документации и за то, что Zend_Mail Framework не поддерживает такую функцию. Приведенные ниже методы работают только для SSL, к сожалению, поскольку TLS включает шифрование после команды EHLO (и многое другое). Если конфигурация действительна, то она возвращает правда, в противном случае он возвращает причину, по которой он неверен в строковой форме.
public function validSMTP () {
// Initialize client configuration, for this example
$config = [
"hostname" => "smtp.gmail.com",
"port" => 465,
"protocol" => "ssl",
"username" => "USERNAME",
"password" => "PASSWORD",
];
// Add the connection url based on protocol
$config ["url"] = $config ["protocol"] . "://" . $config ["hostname"];
// Open a new socket connection to the SMTP mail server
if ( !( $socket = fsockopen ( $config ["url"], $config ["port"], $en, $es, 15 ) ) ) {
// Could not establish connection to host
return "Connection failed, check hostname/port?";
}
// Make sure that the protocol is correct
if ( $this->status_match($socket, '220') === false ) {
// Probably a wrong protocol (SSL/TLS)
return "Connection failed, check protocol?";
}
// Send hello message to server
fwrite ( $socket, "EHLO " . $config ["hostname"] ."\r\n" );
if ( $this->status_match ( $socket, "250" ) === false ) {
// If Hello fails, say config is invalid
return "Invalid SMTP configuration";
}
// Request to login
fwrite ( $socket, "AUTH LOGIN\r\n" );
if ( $this->status_match ( $socket, "334" ) === false ) {
return "Invalid SMTP configuration";
}
// Send the username
fwrite ( $socket, base64_encode ( $config [ "username" ] ) . "\r\n" );
if ( $this->status_match ( $socket, "334" ) === false ) {
// If failed, warn that invalid username password was passed
return "Invalid username/password combination.";
}
// Send the password
fwrite ( $socket, base64_encode ( $config [ "password" ] ) . "\r\n" );
if ( $this->status_match ( $socket, "235" ) === false ) {
// If failed, warn that invalid username password was passed
return "Invalid username/password combination.";
}
// Close the socket connections
fclose ( $socket );
// Return true, if everything above passed
return true;
}
private function status_match ( $socket, $expected ) {
// Initialize the response string
$response = '';
// Get response until nothing but code is visible
while ( substr ( $response, 3, 1) != ' ' ) {
// Receive 250 bytes
if ( !( $response = fgets ( $socket, 256 ) ) ) {
// Break if nothing else to read
break;
}
}
// If the status code is not what was expected
if ( !( substr ( $response, 0, 3 ) == $expected ) ) {
// Return false
return false;
}
// Otherwise return true
return true;
}
Я придумал это, используя следующий пример: http://schoudhury.com/blog/articles/send-email-using-gmail-from-php-with-fsockopen/