SMTP協(xié)議
當(dāng)我們使用PHP的第三方庫或工具類進(jìn)行郵件發(fā)送的時候,是否想過一個問題:
為什么我們不能自己寫php代碼實(shí)現(xiàn)郵件發(fā)現(xiàn),而要用別人的庫呢?php發(fā)送郵件到底是如何實(shí)現(xiàn)的?
首先我們要了解發(fā)送郵件的基本原理,本文基于SMTP協(xié)議實(shí)現(xiàn)郵件發(fā)送
SMTP(Simple Mail Transfer Protocol)即簡單郵件傳輸協(xié)議。簡單來說它定義了一組規(guī)則,我們只需要依照這個規(guī)則來告訴SMTP服務(wù)器,我們要發(fā)送郵件的發(fā)送人,接收人,內(nèi)容,主題等信息。
然后SMTP服務(wù)器依照這組規(guī)則來解析我們發(fā)送的信息,最后進(jìn)行郵件發(fā)送。
像163,qq等郵件服務(wù)器都有提供SMTP服務(wù),我們只要連接上他們的SMTP服務(wù)器,然后write數(shù)據(jù),就能實(shí)現(xiàn)郵件發(fā)送了。
其實(shí)我們可以不寫代碼,直接借用Linux的telnet工具來連接smtp服務(wù),進(jìn)行郵件發(fā)送。借此來了解郵件發(fā)送的整個流程。
telnet進(jìn)行郵件發(fā)送
我們可以在linux環(huán)境下,使用telnet命令,連接163的smtp服務(wù),25端口(一般smtp都是用25端口),借此來理解smtp的傳輸流程。
telnet smtp.163.com 25
然后會得到以下結(jié)果,說明我們連接成功了
Trying 220.181.12.16... Connected to smtp.163.com. Escape character is '^]'. 220 163.com Anti-spam GT for Coremail System (163com[20141201])
接著我們執(zhí)行以下命令,告訴對方我們的身份標(biāo)識來自哪里
HELO smtp.163.com
對方會返回給我們一個250 OK
再執(zhí)行AUTH LOGIN告訴對方我們要開始進(jìn)行身份認(rèn)證,然后對方會回應(yīng)我們一些消息。
后面我們會再輸入我們的用戶名,密碼,發(fā)送郵件的內(nèi)容,發(fā)送人,接受人等信息,然后結(jié)束對話,smtp服務(wù)器就會幫我們把郵件發(fā)送出去。
由于smtp協(xié)議對郵件內(nèi)容格式有嚴(yán)格的要求,在命令行中不好執(zhí)行,所以這里沒有將整個過程執(zhí)行完畢,后面會使用php代碼完整實(shí)現(xiàn)。
從上面使用telnet連接smtp郵件的過程可以看出來,發(fā)送郵件的過程其實(shí)很簡單,就是連接smtp服務(wù)的25端口,依照協(xié)議告訴對方我們要發(fā)什么郵件即可。這與平臺,與編程語言無關(guān)。
無論我們用C語言,還是Java或者PHP,只要使用Socket連接SMTP服務(wù)器,就能實(shí)現(xiàn)郵件發(fā)送。
SMTP指令
上面我們使用telnet連接smtp服務(wù)時,輸入了一些HELO ,AUTH LOGIN等,大家可能會有疑問這些是什么。
其實(shí)很簡單,這些就是SMTP協(xié)議定義的指令,或者說規(guī)則,smtp服務(wù)器就是通過這些指令才知道我們是想干啥。
常用指令如下:
指令 | 作用 |
---|---|
HELO | 向?qū)Ψ洁]件服務(wù)器發(fā)出的標(biāo)識自己的身份的命令 |
AUTH LOGIN | 即將進(jìn)行身份認(rèn)證 |
MAIL FROM | 告訴對方本次郵件發(fā)送人是誰 |
RCPT TO | 發(fā)送給誰 |
DATA | 告訴對方本次郵件,接下來我們發(fā)送郵件具體內(nèi)容了 |
QUIT | 郵件內(nèi)容輸入完畢后,執(zhí)行該指令退出 |
php實(shí)現(xiàn)郵件發(fā)送
直接上代碼
class Mailer { private $host; private $port = 25; private $user; private $pass; private $debug = false; private $sock; public function __construct($host,$port,$user,$pass,$debug = false) { $this->host = $host; $this->port = $port; $this->user = base64_encode($user); //用戶名密碼一定要使用base64編碼才行 $this->pass = base64_encode($pass); $this->debug = $debug; //socket連接 $this->sock = fsockopen($this->host,$this->port); if(!$this->sock){ exit('出錯啦'); } //讀取smtp服務(wù)返回給我們的數(shù)據(jù) $response = fgets($this->sock); $this->debug($response); //如果響應(yīng)中有220返回碼,說明我們連接成功了 if(strstr($response,'220') === false){ exit('出錯啦'); } } //發(fā)送SMTP指令,不同指令的返回碼可能不同 public function execCommand($cmd,$return_code){ fwrite($this->sock,$cmd); $response = fgets($this->sock); //輸出調(diào)試信息 $this->debug('cmd:'.$cmd .';response:'.$response); if(strstr($response,$return_code) === false){ return false; } return true; } public function sendMail($from,$to,$subject,$body){ //detail是郵件的內(nèi)容,一定要嚴(yán)格按照下面的格式,這是協(xié)議規(guī)定的 $detail = 'From:'.$from."\r\n"; $detail .= 'To:'.$to."\r\n"; $detail .= 'Subject:'.$subject."\r\n"; $detail .= 'Content-Type: Text/html;'."\r\n"; $detail .= 'charset=gb2312'."\r\n\r\n"; $detail .= $body; $this->execCommand("HELO ".$this->host."\r\n",250); $this->execCommand("AUTH LOGIN\r\n",334); $this->execCommand($this->user."\r\n",334); $this->execCommand($this->pass."\r\n",235); $this->execCommand("MAIL FROM:<".$from.">\r\n",250); $this->execCommand("RCPT TO:<".$to.">\r\n",250); $this->execCommand("DATA\r\n",354); $this->execCommand($detail."\r\n.\r\n",250); $this->execCommand("QUIT\r\n",221); } public function debug($message){ if($this->debug){ echo '<p>Debug:'.$message . PHP_EOL .'</p>'; } } public function __destruct() { fclose($this->sock); } }