PHP实现支持SSL连接的SMTP邮件发送类


Posted in PHP onMarch 05, 2015

本文实例讲述了PHP实现支持SSL连接的SMTP邮件发送类。分享给大家供大家参考。具体如下:

该实例代码测试过了gmail和QQ邮箱的SMTP。具体代码如下:

<?php

/**

* 邮件发送类

* 支持发送纯文本邮件和HTML格式的邮件,可以多收件人,多抄送,多秘密抄送,带附件(单个或多个附件),支持到服务器的ssl连接

* 需要的php扩展:sockets、Fileinfo和openssl。

* 编码格式是UTF-8,传输编码格式是base64

* @example

* $mail = new MySendMail();

* $mail->setServer("smtp@126.com", "XXXXX@126.com", "XXXXX"); //设置smtp服务器,普通连接方式

* $mail->setServer("smtp.gmail.com", "XXXXX@gmail.com", "XXXXX", 465, true); //设置smtp服务器,到服务器的SSL连接

* $mail->setFrom("XXXXX"); //设置发件人

* $mail->setReceiver("XXXXX"); //设置收件人,多个收件人,调用多次

* $mail->setCc("XXXX"); //设置抄送,多个抄送,调用多次

* $mail->setBcc("XXXXX"); //设置秘密抄送,多个秘密抄送,调用多次

* $mail->addAttachment("XXXX"); //添加附件,多个附件,调用多次

* $mail->setMail("test", "<b>test</b>"); //设置邮件主题、内容

* $mail->sendMail(); //发送

*/

class MySendMail {

    /**

    * @var string 邮件传输代理用户名

    * @access protected

    */

    protected $_userName;

    /**

    * @var string 邮件传输代理密码

    * @access protected

    */

    protected $_password;

    /**

    * @var string 邮件传输代理服务器地址

    * @access protected

    */

    protected $_sendServer;

    /**

    * @var int 邮件传输代理服务器端口

    * @access protected

    */

    protected $_port;

    /**

    * @var string 发件人

    * @access protected

    */

    protected $_from;

    /**

    * @var array 收件人

    * @access protected

    */

    protected $_to = array();

    /**

    * @var array 抄送

    * @access protected

    */

    protected $_cc = array();

    /**

    * @var array 秘密抄送

    * @access protected

    */

    protected $_bcc = array();

    /**

    * @var string 主题

    * @access protected

    */

    protected $_subject;

    /**

    * @var string 邮件正文

    * @access protected

    */

    protected $_body;

    /**

    * @var array 附件

    * @access protected

    */

    protected $_attachment = array();

    /**

    * @var reource socket资源

    * @access protected

    */

    protected $_socket;

    /**

    * @var reource 是否是安全连接

    * @access protected

    */

    protected $_isSecurity;

    /**

    * @var string 错误信息

    * @access protected

    */

    protected $_errorMessage;

    /**

    * 设置邮件传输代理,如果是可以匿名发送有邮件的服务器,只需传递代理服务器地址就行

    * @access public

    * @param string $server 代理服务器的ip或者域名

    * @param string $username 认证账号

    * @param string $password 认证密码

    * @param int $port 代理服务器的端口,smtp默认25号端口

    * @param boolean $isSecurity 到服务器的连接是否为安全连接,默认false

    * @return boolean

    */

    public function setServer($server, $username="", $password="", $port=25, $isSecurity=false) {

        $this->_sendServer = $server;

        $this->_port = $port;

        $this->_isSecurity = $isSecurity;

        $this->_userName = empty($username) ? "" : base64_encode($username);

        $this->_password = empty($password) ? "" : base64_encode($password);

        return true;

    }

    /**

    * 设置发件人

    * @access public

    * @param string $from 发件人地址

    * @return boolean

    */

    public function setFrom($from) {

        $this->_from = $from;

        return true;

    }

    /**

    * 设置收件人,多个收件人,调用多次.

    * @access public

    * @param string $to 收件人地址

    * @return boolean

    */

    public function setReceiver($to) {

        $this->_to[] = $to;

        return true;

    }

    /**

    * 设置抄送,多个抄送,调用多次.

    * @access public

    * @param string $cc 抄送地址

    * @return boolean

    */

    public function setCc($cc) {

        $this->_cc[] = $cc;

        return true;

    }

    /**

    * 设置秘密抄送,多个秘密抄送,调用多次

    * @access public

    * @param string $bcc 秘密抄送地址

    * @return boolean

    */

    public function setBcc($bcc) {

        $this->_bcc[] = $bcc;

        return true;

    }

    /**

    * 设置邮件附件,多个附件,调用多次

    * @access public

    * @param string $file 文件地址

    * @return boolean

    */

    public function addAttachment($file) {

        if(!file_exists($file)) {

            $this->_errorMessage = "file " . $file . " does not exist.";

            return false;

        }

        $this->_attachment[] = $file;

        return true;

    }

    /**

    * 设置邮件信息

    * @access public

    * @param string $body 邮件主题

    * @param string $subject 邮件主体内容,可以是纯文本,也可是是HTML文本

    * @return boolean

    */

    public function setMail($subject, $body) {

        $this->_subject = base64_encode($subject);

        $this->_body = base64_encode($body);

        return true;

    }

    /**

    * 发送邮件

    * @access public

    * @return boolean

    */

    public function sendMail() {

        $command = $this->getCommand();

        $this->_isSecurity ? $this->socketSecurity() : $this->socket();

        foreach ($command as $value) {

            $result = $this->_isSecurity ? $this->sendCommandSecurity($value[0], $value[1]) : $this->sendCommand($value[0], $value[1]);

            if($result) {

                continue;

            }

            else{

                return false;

            }

        }

        //其实这里也没必要关闭,smtp命令:QUIT发出之后,服务器就关闭了连接,本地的socket资源会自动释放

        $this->_isSecurity ? $this->closeSecutity() : $this->close();

        return true;

    }

    /**

    * 返回错误信息

    * @return string

    */

    public function error(){

        if(!isset($this->_errorMessage)) {

            $this->_errorMessage = "";

        }

        return $this->_errorMessage;

    }

    /**

    * 返回mail命令

    * @access protected

    * @return array

    */

    protected function getCommand() {

        $separator = "----=_Part_" . md5($this->_from . time()) . uniqid(); //分隔符

        $command = array(

                array("HELO sendmail\r\n", 250)

            );

        if(!empty($this->_userName)){

            $command[] = array("AUTH LOGIN\r\n", 334);

            $command[] = array($this->_userName . "\r\n", 334);

            $command[] = array($this->_password . "\r\n", 235);

        }

        //设置发件人

        $command[] = array("MAIL FROM: <" . $this->_from . ">\r\n", 250);

        $header = "FROM: <" . $this->_from . ">\r\n";

        //设置收件人

        if(!empty($this->_to)) {

            $count = count($this->_to);

            if($count == 1){

                $command[] = array("RCPT TO: <" . $this->_to[0] . ">\r\n", 250);

                $header .= "TO: <" . $this->_to[0] .">\r\n";

            }

            else{

                for($i=0; $i<$count; $i++){

                    $command[] = array("RCPT TO: <" . $this->_to[$i] . ">\r\n", 250);

                    if($i == 0){

                        $header .= "TO: <" . $this->_to[$i] .">";

                    }

                    elseif($i + 1 == $count){

                        $header .= ",<" . $this->_to[$i] .">\r\n";

                    }

                    else{

                        $header .= ",<" . $this->_to[$i] .">";

                    }

                }

            }

        }

        //设置抄送

        if(!empty($this->_cc)) {

            $count = count($this->_cc);

            if($count == 1){

                $command[] = array("RCPT TO: <" . $this->_cc[0] . ">\r\n", 250);

                $header .= "CC: <" . $this->_cc[0] .">\r\n";

            }

            else{

                for($i=0; $i<$count; $i++){

                    $command[] = array("RCPT TO: <" . $this->_cc[$i] . ">\r\n", 250);

                    if($i == 0){

                    $header .= "CC: <" . $this->_cc[$i] .">";

                    }

                    elseif($i + 1 == $count){

                        $header .= ",<" . $this->_cc[$i] .">\r\n";

                    }

                    else{

                        $header .= ",<" . $this->_cc[$i] .">";

                    }

                }

            }

        }

        //设置秘密抄送

        if(!empty($this->_bcc)) {

            $count = count($this->_bcc);

            if($count == 1) {

                $command[] = array("RCPT TO: <" . $this->_bcc[0] . ">\r\n", 250);

                $header .= "BCC: <" . $this->_bcc[0] .">\r\n";

            }

            else{

                for($i=0; $i<$count; $i++){

                    $command[] = array("RCPT TO: <" . $this->_bcc[$i] . ">\r\n", 250);

                    if($i == 0){

                    $header .= "BCC: <" . $this->_bcc[$i] .">";

                    }

                    elseif($i + 1 == $count){

                        $header .= ",<" . $this->_bcc[$i] .">\r\n";

                    }

                    else{

                        $header .= ",<" . $this->_bcc[$i] .">";

                    }

                }

            }

        }

        //主题

        $header .= "Subject: =?UTF-8?B?" . $this->_subject ."?=\r\n";

        if(isset($this->_attachment)) {

            //含有附件的邮件头需要声明成这个

            $header .= "Content-Type: multipart/mixed;\r\n";

        }

        elseif(false){

            //邮件体含有图片资源的,且包含的图片在邮件内部时声明成这个,如果是引用的远程图片,就不需要了

            $header .= "Content-Type: multipart/related;\r\n";

        }

        else{

            //html或者纯文本的邮件声明成这个

            $header .= "Content-Type: multipart/alternative;\r\n";

        }

        //邮件头分隔符

        $header .= "\t" . 'boundary="' . $separator . '"';

        $header .= "\r\nMIME-Version: 1.0\r\n";

        //这里开始是邮件的body部分,body部分分成几段发送

        $header .= "\r\n--" . $separator . "\r\n";

        $header .= "Content-Type:text/html; charset=utf-8\r\n";

        $header .= "Content-Transfer-Encoding: base64\r\n\r\n";

        $header .= $this->_body . "\r\n";

        $header .= "--" . $separator . "\r\n";

        //加入附件

        if(!empty($this->_attachment)){

            $count = count($this->_attachment);

            for($i=0; $i<$count; $i++){

                $header .= "\r\n--" . $separator . "\r\n";

                $header .= "Content-Type: " . $this->getMIMEType($this->_attachment[$i]) . '; name="=?UTF-8?B?' . base64_encode( basename($this->_attachment[$i]) ) . '?="' . "\r\n";

                $header .= "Content-Transfer-Encoding: base64\r\n";

                $header .= 'Content-Disposition: attachment; filename="=?UTF-8?B?' . base64_encode( basename($this->_attachment[$i]) ) . '?="' . "\r\n";

                $header .= "\r\n";

                $header .= $this->readFile($this->_attachment[$i]);

                $header .= "\r\n--" . $separator . "\r\n";

            }

        }

        //结束邮件数据发送

        $header .= "\r\n.\r\n";

 

        $command[] = array("DATA\r\n", 354);

        $command[] = array($header, 250);

        $command[] = array("QUIT\r\n", 221);

        return $command;

    }

    /**

    * 发送命令

    * @access protected

    * @param string $command 发送到服务器的smtp命令

    * @param int $code 期望服务器返回的响应吗

    * @return boolean

    */

    protected function sendCommand($command, $code) {

        echo 'Send command:' . $command . ',expected code:' . $code . '<br />';

        //发送命令给服务器

        try{

            if(socket_write($this->_socket, $command, strlen($command))){

                //当邮件内容分多次发送时,没有$code,服务器没有返回

                if(empty($code))  {

                    return true;

                }

                //读取服务器返回

                $data = trim(socket_read($this->_socket, 1024));

                echo 'response:' . $data . '<br /><br />';

                if($data) {

                    $pattern = "/^".$code."+?/";

                    if(preg_match($pattern, $data)) {

                        return true;

                    }

                    else{

                        $this->_errorMessage = "Error:" . $data . "|**| command:";

                        return false;

                    }

                }

                else{

                    $this->_errorMessage = "Error:" . socket_strerror(socket_last_error());

                    return false;

                }

            }

            else{

                $this->_errorMessage = "Error:" . socket_strerror(socket_last_error());

                return false;

            }

        }catch(Exception $e) {

            $this->_errorMessage = "Error:" . $e->getMessage();

        }

    }

    /**

    * 安全连接发送命令

    * @access protected

    * @param string $command 发送到服务器的smtp命令

    * @param int $code 期望服务器返回的响应吗

    * @return boolean

    */

    protected function sendCommandSecurity($command, $code) {

        echo 'Send command:' . $command . ',expected code:' . $code . '<br />';

        try {

            if(fwrite($this->_socket, $command)){

                //当邮件内容分多次发送时,没有$code,服务器没有返回

                if(empty($code))  {

                    return true;

                }

                //读取服务器返回

                $data = trim(fread($this->_socket, 1024));

                echo 'response:' . $data . '<br /><br />';

                if($data) {

                    $pattern = "/^".$code."+?/";

                    if(preg_match($pattern, $data)) {

                        return true;

                    }

                    else{

                        $this->_errorMessage = "Error:" . $data . "|**| command:";

                        return false;

                    }

                }

                else{

                    return false;

                }

            }

            else{

                $this->_errorMessage = "Error: " . $command . " send failed";

                return false;

            }

        }catch(Exception $e) {

            $this->_errorMessage = "Error:" . $e->getMessage();

        }

    } 

    /**

    * 读取附件文件内容,返回base64编码后的文件内容

    * @access protected

    * @param string $file 文件

    * @return mixed

    */

    protected function readFile($file) {

        if(file_exists($file)) {

            $file_obj = file_get_contents($file);

            return base64_encode($file_obj);

        }

        else {

            $this->_errorMessage = "file " . $file . " dose not exist";

            return false;

        }

    }

    /**

    * 获取附件MIME类型

    * @access protected

    * @param string $file 文件

    * @return mixed

    */

    protected function getMIMEType($file) {

        if(file_exists($file)) {

            $mime = mime_content_type($file);

            /*if(! preg_match("/gif|jpg|png|jpeg/", $mime)){

                $mime = "application/octet-stream";

            }*/

            return $mime;

        }

        else {

            return false;

        }

    }

    /**

    * 建立到服务器的网络连接

    * @access protected

    * @return boolean

    */

    protected function socket() {

        //创建socket资源

        $this->_socket = socket_create(AF_INET, SOCK_STREAM, getprotobyname('tcp'));

        if(!$this->_socket) {

            $this->_errorMessage = socket_strerror(socket_last_error());

            return false;

        }

        socket_set_block($this->_socket);//设置阻塞模式

        //连接服务器

        if(!socket_connect($this->_socket, $this->_sendServer, $this->_port)) {

            $this->_errorMessage = socket_strerror(socket_last_error());

            return false;

        }

        $str = socket_read($this->_socket, 1024);

        if(!preg_match("/220+?/", $str)){

            $this->_errorMessage = $str;

            return false;

        }

        return true;

    }

    /**

    * 建立到服务器的SSL网络连接

    * @access protected

    * @return boolean

    */

    protected function socketSecurity() {

        $remoteAddr = "tcp://" . $this->_sendServer . ":" . $this->_port;

        $this->_socket = stream_socket_client($remoteAddr, $errno, $errstr, 30);

        if(!$this->_socket){

            $this->_errorMessage = $errstr;

            return false;

        }

        //设置加密连接,默认是ssl,如果需要tls连接,可以查看php手册stream_socket_enable_crypto函数的解释

        stream_socket_enable_crypto($this->_socket, true, STREAM_CRYPTO_METHOD_SSLv23_CLIENT);

        stream_set_blocking($this->_socket, 1); //设置阻塞模式

        $str = fread($this->_socket, 1024);

        if(!preg_match("/220+?/", $str)){

            $this->_errorMessage = $str;

            return false;

        }

        return true;

    }

    /**

    * 关闭socket

    * @access protected

    * @return boolean

    */

    protected function close() {

        if(isset($this->_socket) && is_object($this->_socket)) {

            $this->_socket->close();

            return true;

        }

        $this->_errorMessage = "No resource can to be close";

        return false;

    }

    /**

    * 关闭安全socket

    * @access protected

    * @return boolean

    */

    protected function closeSecutity() {

        if(isset($this->_socket) && is_object($this->_socket)) {

            stream_socket_shutdown($this->_socket, STREAM_SHUT_WR);

            return true;

        }

        $this->_errorMessage = "No resource can to be close";

        return false;

    }

}

希望本文所述对大家的php程序设计有所帮助。

PHP 相关文章推荐
MySql中正则表达式的使用方法描述
Jul 30 PHP
快速开发一个PHP扩展图文教程
Dec 12 PHP
phpmyadmin导入(import)文件限制的解决办法
Dec 11 PHP
php 生成随机验证码图片代码
Feb 08 PHP
PHP URL地址获取函数代码(端口等) 推荐
May 15 PHP
PHP XML数据解析代码
May 26 PHP
ThinkPHP CURD方法之limit方法详解
Jun 18 PHP
PHP实现生成透明背景的PNG缩略图函数分享
Jul 08 PHP
学习php设计模式 php实现装饰器模式(decorator)
Dec 07 PHP
PHP 芝麻信用接入的注意事项
Dec 01 PHP
PHP实现动态添加XML中数据的方法
Mar 30 PHP
MacOS下PHP7.1升级到PHP7.4.15的方法
Feb 22 PHP
PHP 使用redis简单示例分享
Mar 05 #PHP
memcache一致性hash的php实现方法
Mar 05 #PHP
PHP将session信息存储到数据库的类实例
Mar 04 #PHP
php微信支付之APP支付方法
Mar 04 #PHP
php支付宝手机网页支付类实例
Mar 04 #PHP
php银联网页支付实现方法
Mar 04 #PHP
php随机抽奖实例分析
Mar 04 #PHP
You might like
php5 mysql分页实例代码
2008/04/10 PHP
一个基于PDO的数据库操作类(新) 一个PDO事务实例
2011/07/03 PHP
php中替换字符串中的空格为逗号','的方法
2014/06/09 PHP
Thinkphp将二维数组变为标签适用的一维数组方法总结
2014/10/30 PHP
php文件管理基本功能简单操作
2017/01/16 PHP
PHP编程中的Session阻塞问题与解决方法分析
2017/08/07 PHP
php+mysql开发中的经验与常识小结
2019/03/25 PHP
JavaScript 基础篇之运算符、语句(二)
2012/04/07 Javascript
js相册效果代码(点击创建即可)
2013/04/16 Javascript
原生javascript兼容性测试实例
2013/07/01 Javascript
JS中实现replaceAll的方法(实例代码)
2013/11/12 Javascript
jQuery实现带有洗牌效果的动画分页实例
2015/08/31 Javascript
理解JS事件循环
2016/01/07 Javascript
详解vue数据渲染出现闪烁问题
2017/06/29 Javascript
详解让sublime text3支持Vue语法高亮显示的示例
2017/09/29 Javascript
Nodejs下使用gm圆形裁剪并合成图片的示例
2018/02/22 NodeJs
JavaScript 实现HTML DOM增删改查操作的常见方法详解
2020/01/04 Javascript
Node.js API详解之 module模块用法实例分析
2020/05/13 Javascript
Vue实现手机号、验证码登录(60s禁用倒计时)
2020/12/19 Vue.js
Python用Pillow(PIL)进行简单的图像操作方法
2017/07/07 Python
PyQt5每天必学之事件与信号
2018/04/20 Python
神经网络相关之基础概念的讲解
2018/12/29 Python
Python 中判断列表是否为空的方法
2019/11/24 Python
python 实现批量替换文本中的某部分内容
2019/12/13 Python
使用Python获取当前工作目录和执行命令的位置
2020/03/09 Python
TensorFlow tf.nn.softmax_cross_entropy_with_logits的用法
2020/04/19 Python
使用pytorch实现论文中的unet网络
2020/06/24 Python
CSS3实现歌词进度文字颜色填充变化动态效果的思路详解
2020/06/02 HTML / CSS
英国珠宝钟表和家居礼品精品店:David Shuttle
2018/02/24 全球购物
房产委托公证书
2014/04/08 职场文书
户外宣传策划方案
2014/05/25 职场文书
大学生迟到检讨书500字
2014/10/17 职场文书
承诺保证书格式
2015/02/28 职场文书
开学典礼校长致辞
2015/07/29 职场文书
2016银行求职自荐信
2016/01/28 职场文书
【海涛dota解说】海涛小满开黑4v5被破两路翻盘潮汐第一视角解说
2022/04/01 DOTA