用BCB開(kāi)發(fā)有身份認(rèn)證技巧的Email程序
發(fā)表時(shí)間:2024-06-12 來(lái)源:明輝站整理相關(guān)軟件相關(guān)文章人氣:
[摘要]引言 為了更有效地抑制垃圾郵件的泛濫,目前多數(shù)網(wǎng)站的郵件收發(fā)系統(tǒng)都使用了ESMTP服務(wù)的身份認(rèn)證功能。即用戶(hù)發(fā)送郵件時(shí),需要對(duì)用戶(hù)的身份進(jìn)行驗(yàn)證,如果帳號(hào)或密碼錯(cuò)誤,郵件服務(wù)器會(huì)拒絕發(fā)送郵件。Borland C++ Builder 6中有豐富的控件供開(kāi)發(fā)者使用,其中當(dāng)然也包括郵件發(fā)送控件NMSM...
引言
為了更有效地抑制垃圾郵件的泛濫,目前多數(shù)網(wǎng)站的郵件收發(fā)系統(tǒng)都使用了ESMTP服務(wù)的身份認(rèn)證功能。即用戶(hù)發(fā)送郵件時(shí),需要對(duì)用戶(hù)的身份進(jìn)行驗(yàn)證,如果帳號(hào)或密碼錯(cuò)誤,郵件服務(wù)器會(huì)拒絕發(fā)送郵件。Borland C++ Builder 6中有豐富的控件供開(kāi)發(fā)者使用,其中當(dāng)然也包括郵件發(fā)送控件NMSMTP,這個(gè)控件使用方便,但是惟一的缺點(diǎn)是不支持郵件發(fā)送時(shí)的身份認(rèn)證功能。筆者通過(guò)對(duì)郵件發(fā)送協(xié)議的分析,在使用控件的基礎(chǔ)上設(shè)計(jì)了具有身份認(rèn)證功能的郵件發(fā)送程序。
ESMTP協(xié)議分析
為了實(shí)現(xiàn)身份認(rèn)證功能,目前ESMTP協(xié)議中增加了一部分內(nèi)容,這就是身份認(rèn)證。下面我們看看這段認(rèn)證過(guò)程,以筆者在網(wǎng)易的郵箱為例(其中C表示客戶(hù)端,S表示郵件服務(wù)器):
(1)C: AUTH LOGIN
(2)S: 334 dXNlcm5hbWU6
(3)C: d3lxX2puX3NkX2Nu
(4)S: 334 UGFzc3dvcmQ6
(5)C: 密碼略去
(6)S: 235 Authentication successful
詳細(xì)說(shuō)明:
(1)客戶(hù)端向服務(wù)器發(fā)送認(rèn)證指令。
(2)服務(wù)器返回Base64編碼串,334意味成功。編碼字符串解碼后為"username:",說(shuō)明要求客戶(hù)端發(fā)送用戶(hù)名。
(3)客戶(hù)端發(fā)送Base64編碼的用戶(hù)名串,此處為"wyq_jn_sd_cn"。
(4)服務(wù)器返回Base64編碼串,334意味成功。編碼字符串解碼后為"password:",說(shuō)明要求客戶(hù)端發(fā)送用戶(hù)口令。
(5)客戶(hù)端發(fā)送Base64編碼的口令串,此處略去。
(6)服務(wù)器返回普通字符串,235意味成功,表示認(rèn)證成功可以發(fā)送郵件了。
MIME Base64編碼解釋
一般的計(jì)算機(jī)編碼的一個(gè)字節(jié)是8bit,0——FF就是256種不同的8bit組合。我們現(xiàn)在要介紹的這種Base64編碼則是每個(gè)字節(jié)6bit,共有26=64種組合。其中每種組合對(duì)應(yīng)一個(gè)字符,這些字符是“ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz01234567 89+/!边@就意味著每3個(gè)普通編碼可以轉(zhuǎn)換成4個(gè)Base64編碼,那么如果需要轉(zhuǎn)換的普通編碼不是3的整數(shù)倍怎么辦?Base64規(guī)定,位數(shù)不足的字節(jié)后面補(bǔ)0,然后差幾個(gè)字符補(bǔ)幾個(gè)‘=’號(hào)。
設(shè)計(jì)思路
我們可以使用NMSMTP控件與郵件服務(wù)器連接。通過(guò)調(diào)用Connect方法,然后監(jiān)聽(tīng)OnConnect事件;在OnConnect事件里我們可以增加身份認(rèn)證功能。這里是主要利用了NMSMTP從Powersock中繼承的一些基本網(wǎng)絡(luò)通訊函數(shù),包括Read,DataAvailable,SendBuffer等來(lái)實(shí)現(xiàn)身份認(rèn)證過(guò)程。如果身份認(rèn)證成功,就可以繼續(xù)進(jìn)行郵件發(fā)送;否則,提示錯(cuò)誤信息,斷開(kāi)網(wǎng)絡(luò)連接。
程序?qū)崿F(xiàn)
使用BCB設(shè)計(jì)如圖1所示的窗體。
圖1 程序主界面
1、在登錄按鈕的OnClick事件中調(diào)用連接函數(shù)
void __fastcall TForm1::Logon1Click(TObject *Sender)
{
AddLog("正在登錄"+Edit1->Text+"......");
NMSMTP1->Host = Edit1->Text; //主機(jī)地址
NMSMTP1->Port = 25; //主機(jī)端口,缺省為25
NMSMTP1->UserID = Edit4->Text; //用戶(hù)名
NMSMTP1->Connect(); //連接主機(jī)
}
2、處理OnConnect事件
void __fastcall TForm1::NMSMTP1Connect(TObject *Sender)
{
AddLog("連接服務(wù)器成功。");
AnsiString Data="",rData="";
bool b_ok;
if(CheckBox1->Checked){
Data="AUTH LOGIN\r\n"; //登錄請(qǐng)求命令
NMSMTP1->SendBuffer(Data.c_str(),Data.Length()); //命令發(fā)出
rData = WaitForReply(5); //等待接收返回?cái)?shù)據(jù),5秒內(nèi)必須返回
b_ok = false;
if(rData.Length()>=3){
//334意味著服務(wù)器要求輸入用戶(hù)名
if(rData.TrimLeft().SubString(0,3)=="334"){
AddLog("正在驗(yàn)證身份......");
b_ok =true;
}
}
if(!b_ok){
AddLog("登錄失敗,正在退出......");
NMSMTP1->Disconnect();
return;
}
rData="";
Data=encode(Edit4->Text)+"\r\n"; //用戶(hù)名轉(zhuǎn)換為Base64編碼。
NMSMTP1->SendBuffer(Data.c_str(),Data.Length()); //發(fā)送用戶(hù)名
rData = WaitForReply(5);
b_ok=false;
if(rData.Length()>=3){
// 334意味著服務(wù)器要求輸入口令
if(rData.TrimLeft().SubString(0,3)=="334"){
AddLog("正在驗(yàn)證口令......");
b_ok =true;
}
}
if(!b_ok){
AddLog("登錄失敗,正在退出......");
NMSMTP1->Disconnect();
return;
}
rData="";
Data=encode(Edit5->Text)+"\r\n";//口令轉(zhuǎn)換成Base64編碼。
NMSMTP1->SendBuffer(Data.c_str(),Data.Length()); //發(fā)送口令
rData=WaitForReply(5);
b_ok = false;
if(rData.Length()>=3){
if(rData.TrimLeft().SubString(0,3)=="235"){
AddLog("登錄成功......");
b_ok =true;
}
}
if(!b_ok){
AddLog("登錄失敗,正在退出......");
NMSMTP1->Disconnect();
return;
}
}
SendMail->Enabled=true; //允許發(fā)送郵件
disconnect->Enabled=true; //允許斷開(kāi)連接
Logon1->Enabled=false; //不允許再次登錄
}
3、MIME Base64編碼轉(zhuǎn)換
AnsiString TForm1::encode(AnsiString s)
{
int m_len; //字符串長(zhǎng)度
int i; //循環(huán)變量
int m_tmp; //臨時(shí)變量
AnsiString m_64code; //儲(chǔ)存Base64編碼的字符串
char* m_s; //臨時(shí)存儲(chǔ)參數(shù)字符串
//Base64字符表
char m_64[]= "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
m_len = s.Length(); //取得字符串長(zhǎng)度
m_s = s.c_str();
m_64code=""; //返回串置空
//處理3的倍數(shù)以?xún)?nèi)的字符
for(i=0;i<m_len-m_len%3;i+=3){
m_tmp=m_s[i]/4;
m_64code+=m_64[m_tmp];
m_tmp=m_s[i]%4*16 + m_s[i+1]/16;
m_64code+=m_64[m_tmp];
m_tmp=m_s[i+1]%16*4 + m_s[i+2]/64;
m_64code+=m_64[m_tmp];
m_tmp=m_s[i+2]%64;
m_64code+=m_64[m_tmp];
}
//如果字符串的長(zhǎng)度被3除余2 ,不足的位數(shù)補(bǔ)0,尾部補(bǔ)“=”
if(m_len%3==2){
m_tmp=m_s[m_len-2]/4;
m_64code+=m_64[m_tmp];
m_tmp=m_s[m_len-2]%4*16+m_s[m_len-1]/16;
m_64code+=m_64[m_tmp];
m_tmp=m_s[m_len-1]%16*4;
m_64code+=m_64[m_tmp];
m_64code+='=';
}
//如果字符串的長(zhǎng)度被3除余1 ,不足的位數(shù)補(bǔ)0,尾部補(bǔ)兩個(gè)“=”
if(m_len%3==1){
m_tmp=m_s[m_len-1]/4;
m_64code+=m_64[m_tmp];
m_tmp=m_s[m_len-1]%4*16;
m_64code+=m_64[m_tmp];
m_64code+="==";
}
return m_64code;
}
結(jié)束語(yǔ)
本程序在Windows 2000環(huán)境下使用Borland C++ Builder 6.0編寫(xiě)及調(diào)試的,分別使用網(wǎng)易和新浪郵箱做實(shí)驗(yàn),都可以順利完成身份認(rèn)證以及郵件發(fā)送功能。