using System;
using System.IO;
using System.Text;
using System.Net;
using System.Net.Sockets;
using System.Collections;
using System.Collections.Specialized;
using KSN.Exceptions;
using KSN.Validate;
namespace KSN.Web.Mail
{
/// <summary>
/// 郵件內容
/// </summary>
public class MailMessage
{
private string sender=null;
private StringCollection receivers=new StringCollection();
private string subject="";
private string xMailer="";
private StringCollection attachments=new StringCollection();
private MailEncodings mailEncoding=MailEncodings.GB2312;
private MailTypes mailType=MailTypes.Html;
private byte[] mailBody=null;
/// <summary>
/// 獲取或設置發件人
/// </summary>
public string Sender
{
get{return this.sender;}
set{this.sender=value;}
}
/// <summary>
/// 獲取收件人地址集合
/// </summary>
public StringCollection Receivers
{
get{return this.receivers;}
}
/// <summary>
/// 獲取或設置郵件主題
/// </summary>
public string Subject
{
get{return this.subject;}
set{this.subject=value;}
}
/// <summary>
/// 獲取或設置郵件傳送者
/// </summary>
public string XMailer
{
get{return this.xMailer;}
set{this.xMailer=value;}
}
/// <summary>
/// 獲取附件列表
/// </summary>
public StringCollection Attachments
{
get{return this.attachments;}
}
/// <summary>
/// 獲取或設置郵件的編碼方式
/// </summary>
public MailEncodings MailEncoding
{
get{return this.mailEncoding;}
set{this.mailEncoding=value;}
}
/// <summary>
/// 獲取或設置郵件格式
/// </summary>
public MailTypes MailType
{
get{return this.mailType;}
set{this.mailType=value;}
}
/// <summary>
/// 獲取或設置郵件正文
/// </summary>
public byte[] MailBody
{
get{return this.mailBody;}
set{this.mailBody=value;}
}
}
/// <summary>
/// 郵件編碼
/// </summary>
public enum MailEncodings
{
GB2312,
ASCII,
Unicode,
UTF8
}
/// <summary>
/// 郵件格式
/// </summary>
public enum MailTypes
{
Html,
Text
}
/// <summary>
/// smtp服務器的驗證方式
/// </summary>
public enum SmtpValidateTypes
{
/// <summary>
/// 不需要驗證
/// </summary>
None,
/// <summary>
/// 通用的auth login驗證
/// </summary>
Login,
/// <summary>
/// 通用的auth plain驗證
/// </summary>
Plain,
/// <summary>
/// CRAM-MD5驗證
/// </summary>
CRAMMD5
}
/// <summary>
/// 郵件發送類
/// </summary>
public class KSN_Smtp
{
#region "member fIElds"
/// <summary>
/// 連接對象
/// </summary>
private TcpClIEnt tc;
/// <summary>
/// 網絡流
/// </summary>
private NetworkStream ns;
/// <summary>
/// 錯誤的代碼字典
/// </summary>
private StringDictionary errorCodes=new StringDictionary();
/// <summary>
/// 操作執行成功後的響應代碼字典
/// </summary>
private StringDictionary rightCodes=new StringDictionary();
/// <summary>
/// 執行過程中錯誤的消息
/// </summary>
private string errorMessage="";
/// <summary>
/// 記錄操作日志
/// </summary>
private string logs="";
/// <summary>
/// 主機登陸的驗證方式
/// </summary>
private StringCollection validateTypes=new StringCollection();
/// <summary>
/// 換行常數
/// </summary>
private const string CRLF="\r\n";
private string serverName="smtp";
private string logPath=null;
private string userid=null;
private string passWord=null;
private string mailEncodingName="GB2312";
private bool sendIsComplete=false;
private SmtpValidateTypes smtpValidateType=SmtpValidateTypes.Login;
#endregion
#region "propertys"
/// <summary>
/// 獲取最後一此程序執行中的錯誤消息
/// </summary>
public string ErrorMessage
{
get{return this.errorMessage;}
}
/// <summary>
/// 獲取或設置日志輸出路徑
/// </summary>
public string LogPath
{
get
{
return this.logPath;
}
set{this.logPath=value;}
}
/// <summary>
/// 獲取或設置登陸smtp服務器的帳號
/// </summary>
public string UserID
{
get{return this.userid;}
set{this.userid=value;}
}
/// <summary>
/// 獲取或設置登陸smtp服務器的密碼
/// </summary>
public string PassWord
{
get{return this.passWord;}
set{this.passWord=value;}
}
/// <summary>
/// 獲取或設置要使用登陸Smtp服務器的驗證方式
/// </summary>
public SmtpValidateTypes SmtpValidateType
{
get{return this.smtpValidateType;}
set{this.smtpValidateType=value;}
}
#endregion
#region "construct functions"
/// <summary>
/// 構造函數
/// </summary>
/// <param name="server">主機名</param>
/// <param name="port">端口</param>
public KSN_Smtp(string server,int port)
{
tc=new TcpClIEnt(server,port);
ns=tc.GetStream();
this.serverName=server;
this.initialFIElds();
}
/// <summary>
/// 構造函數
/// </summary>
/// <param name="ip">主機ip</param>
/// <param name="port">端口</param>
public KSN_Smtp(IPAddress ip,int port)
{
IPEndPoint endPoint=new IPEndPoint(ip,port);
tc=new TcpClIEnt(endPoint);
ns=tc.GetStream();
this.serverName=ip.ToString();
this.initialFIElds();
}
#endregion
#region "methods"
private void initialFIElds() //初始化連接
{
logs="================"+DateTime.Now.ToLongDateString()+" "+DateTime.Now.ToLongTimeString()+"==============="+CRLF;
//*****************************************************************
//錯誤的狀態碼
//*****************************************************************
errorCodes.Add("421","服務未就緒,關閉傳輸通道");
errorCodes.Add("432","需要一個密碼轉換");
errorCodes.Add("450","要求的郵件操作未完成,郵箱不可用(如:郵箱忙)");
errorCodes.Add("451","放棄要求的操作,要求的操作未執行");
errorCodes.Add("452","系統存儲不足,要求的操作未完成");
errorCodes.Add("454","臨時的認證失敗");
errorCodes.Add("500","郵箱地址錯誤");
errorCodes.Add("501","參數格式錯誤");
errorCodes.Add("502","命令不可實現");
errorCodes.Add("503","命令的次序不正確");
errorCodes.Add("504","命令參數不可實現");
errorCodes.Add("530","需要認證");
errorCodes.Add("534","認證機制過於簡單");
errorCodes.Add("538","當前請求的認證機制需要加密");
errorCodes.Add("550","當前的郵件操作未完成,郵箱不可用(如:郵箱未找到或郵箱不能用)");
errorCodes.Add("551","用戶非本地,請嘗試<forward-path>");
errorCodes.Add("552","過量的存儲分配,制定的操作未完成");
errorCodes.Add("553","郵箱名不可用,如:郵箱地址的格式錯誤");
errorCodes.Add("554","傳送失敗");
errorCodes.Add("535","用戶身份驗證失敗");
//****************************************************************
//操作執行成功後的狀態碼
//****************************************************************
rightCodes.Add("220","服務就緒");
rightCodes.Add("221","服務關閉傳輸通道");
rightCodes.Add("235","驗證成功");
rightCodes.Add("250","要求的郵件操作完成");
rightCodes.Add("251","非本地用戶,將轉發向<forward-path>");
rightCodes.Add("334","服務器響應驗證Base64字符串");
rightCodes.Add("354","開始郵件輸入,以<CRLF>.<CRLF>結束");
//讀取系統回應
StreamReader reader=new StreamReader(ns);
logs+=reader.ReadLine()+CRLF;
}
/// <summary>
/// 向SMTP發送命令
/// </summary>
/// <param name="cmd"></param>
private string sendCommand(string cmd,bool isMailData)
{
if(cmd!=null && cmd.Trim()!=string.Empty)
{
byte[] cmd_b=null;
if(!isMailData)//不是郵件數據
cmd+=CRLF;
logs+=cmd;
//開始寫入郵件數據
if(!isMailData)
{
cmd_b=Encoding.ASCII.GetBytes(cmd);
ns.Write(cmd_b,0,cmd_b.Length);
}
else
{
cmd_b=Encoding.GetEncoding(this.mailEncodingName).GetBytes(cmd);
ns.BeginWrite(cmd_b,0,cmd_b.Length,new AsyncCallback(this.asyncCallBack),null);
}
//讀取服務器響應
StreamReader reader=new StreamReader(ns);
string response=reader.ReadLine();
logs+=response+CRLF;
//檢查狀態碼
string statusCode=response.Substring(0,3);
bool isExist=false;
bool isRightCode=true;
foreach(string err in this.errorCodes.Keys)
{
if(statusCode==err)
{
isExist=true;
isRightCode=false;
break;
}
}
foreach(string right in this.rightCodes.Keys)
{
if(statusCode==right)
{
isExist=true;
break;
}
}
//根據狀態碼來處理下一步的動作
if(!isExist) //不是合法的SMTP主機
{
this.setError("不是合法的SMTP主機,或服務器拒絕服務");
}
else if(!isRightCode)//命令沒能成功執行
{
this.setError(statusCode+":"+this.errorCodes[statusCode]);
}
else //命令成功執行
{
this.errorMessage="";
}
return response;
}
else
{
return null;
}
}
/// <summary>
/// 通過auth login方式登陸smtp服務器
/// </summary>
private void landingByLogin()
{
string base64UserId=this.convertBase64String(this.UserID,"ASCII");
string base64Pass=this.convertBase64String(this.PassWord,"ASCII");
//握手
this.sendCommand("helo "+this.serverName,false);
//開始登陸
this.sendCommand("auth login",false);
this.sendCommand(base64UserId,false);
this.sendCommand(base64Pass,false);
}
/// <summary>
/// 通過auth plain方式登陸服務器
/// </summary>
private void landingByPlain()
{
string NULL=((char)0).ToString();
string loginStr=NULL+this.UserID+NULL+this.PassWord;
string base64LoginStr=this.convertBase64String(loginStr,"ASCII");
//握手
this.sendCommand("helo "+this.serverName,false);
//登陸
this.sendCommand(base64LoginStr,false);
}
/// <summary>
/// 通過auth CRAM-MD5方式登陸
/// </summary>
private void landingByCRAMMD5()
{
//握手
this.sendCommand("helo "+this.serverName,false);
//登陸
string response=this.sendCommand("auth CRAM-MD5",false);
//得到服務器返回的標識
string identifIEr=response.Remove(0,4);
//用MD5加密標識
KSN_MACTripleDES mac=new KSN_MacTripleDES();
Mac.Key=this.PassWord;
Mac.Data=Encoding.ASCII.GetBytes(identifIEr);
string hash=Mac.GetHashValue();
//發送用戶帳號信息
this.sendCommand(this.UserID+" "+hash,false);
}
/// <summary>
/// 發送郵件
/// </summary>
/// <returns></returns>
public bool SendMail(MailMessage mail)
{
bool isSended=true;
try
{
//檢測發送郵件的必要條件
if(mail.Sender==null)
{
this.setError("沒有設置發信人");
}
if(mail.Receivers.Count==0)
{
this.setError("至少要有一個收件人");
}
if(this.SmtpValidateType!=SmtpValidateTypes.None)
{
if(this.userid==null || this.passWord==null)
{
this.setError("當前設置需要smtp驗證,但是沒有給出登陸帳號");
}
}
//開始登陸
switch(this.SmtpValidateType)
{
case SmtpValidateTypes.None:
this.sendCommand("helo "+this.serverName,false);
break;
case SmtpValidateTypes.Login:
this.landingByLogin();
break;
case SmtpValidateTypes.Plain:
this.landingByPlain();
break;
case SmtpValidateTypes.CRAMMD5:
this.landingByCRAMMD5();
break;
default:
break;
}
//初始化郵件會話(對應SMTP命令mail)
this.sendCommand("mail from:<"+mail.Sender+">",false);
//標識收件人(對應SMTP命令Rcpt)
foreach(string receive in mail.Receivers)
{
this.sendCommand("rcpt to:<"+receive+">",false);
}
//標識開始輸入郵件內容(Data)
this.sendCommand("data",false);
//開始編寫郵件內容
string message="Subject:"+mail.Subject+CRLF;
message+="X-mailer:"+mail.XMailer+CRLF;
message+="MIME-Version:1.0"+CRLF;
if(mail.Attachments.Count==0)//沒有附件
{
if(mail.MailType==MailTypes.Text) //文本格式
{
message+="Content-Type:text/plain;"+CRLF+" ".PadRight(8,' ')+"charset=\""+
mail.MailEncoding.ToString()+"\""+CRLF;
message+="Content-Transfer-Encoding:base64"+CRLF+CRLF;
if(mail.MailBody!=null)
message+=Convert.ToBase64String(mail.MailBody,0,mail.MailBody.Length)+CRLF+CRLF+CRLF+"."+CRLF;
}
else//Html格式
{
message+="Content-Type:multipart/alertnative;"+CRLF+" ".PadRight(8,' ')+"boundary"
+"=\"=====003_Dragon310083331177_=====\""+CRLF+CRLF+CRLF;
message+="This is a multi-part message in MIME format"+CRLF+CRLF;
message+="--=====003_Dragon310083331177_====="+CRLF;
message+="Content-Type:text/Html;"+CRLF+" ".PadRight(8,' ')+"charset=\""+
mail.MailEncoding.ToString().ToLower()+"\""+CRLF;
message+="Content-Transfer-Encoding:base64"+CRLF+CRLF;
if(mail.MailBody!=null)
message+=Convert.ToBase64String(mail.MailBody,0,mail.MailBody.Length)+CRLF+CRLF;
message+="--=====003_Dragon310083331177_=====--"+CRLF+CRLF+CRLF+"."+CRLF;
}
}
else//有附件
{
//處理要在郵件中顯示的每個附件的數據
StringCollection attatchmentDatas=new StringCollection();
foreach(string path in mail.Attachments)
{
if(!File.Exists(path))
{
this.setError("指定的附件沒有找到"+path);
}
else
{
//得到附件的字節流
FileInfo file=new FileInfo(path);
FileStream fs=new FileStream(path,FileMode.Open,FileAccess.Read);
if(fs.Length>(long)int.MaxValue)
{
this.setError("附件的大小超出了最大限制");
}
byte[] file_b=new byte[(int)fs.Length];
fs.Read(file_b,0,file_b.Length);
fs.Close();
string attatchmentMailStr="Content-Type:application/octet-stream;"+CRLF+" ".PadRight(8,' ')+"name="+
"\""+file.Name+"\""+CRLF;
attatchmentMailStr+="Content-Transfer-Encoding:base64"+CRLF;
attatchmentMailStr+="Content-Disposition:attachment;"+CRLF+" ".PadRight(8,' ')+"filename="+
"\""+file.Name+"\""+CRLF+CRLF;
attatchmentMailStr+=Convert.ToBase64String(file_b,0,file_b.Length)+CRLF+CRLF;
attatchmentDatas.Add(attatchmentMailStr);
}
}
//設置郵件信息
if(mail.MailType==MailTypes.Text) //文本格式
{
message+="Content-Type:multipart/mixed;"+CRLF+" ".PadRight(8,' ')+"boundary=\"=====001_Dragon320037612222_=====\""
+CRLF+CRLF;
message+="This is a multi-part message in MIME format."+CRLF+CRLF;
message+="--=====001_Dragon320037612222_====="+CRLF;
message+="Content-Type:text/plain;"+CRLF+" ".PadRight(8,' ')+"charset=\""+mail.MailEncoding.ToString().ToLower()+"\""+CRLF;
message+="Content-Transfer-Encoding:base64"+CRLF+CRLF;
if(mail.MailBody!=null)
message+=Convert.ToBase64String(mail.MailBody,0,mail.MailBody.Length)+CRLF;
foreach(string s in attatchmentDatas)
{
message+="--=====001_Dragon320037612222_====="+CRLF+s+CRLF+CRLF;
}
message+="--=====001_Dragon320037612222_=====--"+CRLF+CRLF+CRLF+"."+CRLF;
}
else
{
message+="Content-Type:multipart/mixed;"+CRLF+" ".PadRight(8,' ')+"boundary=\"=====001_Dragon255511664284_=====\""
+CRLF+CRLF;
message+="This is a multi-part message in MIME format."+CRLF+CRLF;
message+="--=====001_Dragon255511664284_====="+CRLF;
message+="Content-Type:text/Html;"+CRLF+" ".PadRight(8,' ')+"charset=\""+mail.MailEncoding.ToString().ToLower()+"\""+CRLF;
message+="Content-Transfer-Encoding:base64"+CRLF+CRLF;
if(mail.MailBody!=null)
message+=Convert.ToBase64String(mail.MailBody,0,mail.MailBody.Length)+CRLF+CRLF;
for(int i=0;i<attatchmentDatas.Count;i++)
{
message+="--=====001_Dragon255511664284_====="+CRLF+attatchmentDatas[i]+CRLF+CRLF;
}
message+="--=====001_Dragon255511664284_=====--"+CRLF+CRLF+CRLF+"."+CRLF;
}
}
//發送郵件數據
this.mailEncodingName=mail.MailEncoding.ToString();
this.sendCommand(message,true);
if(this.sendIsComplete)
this.sendCommand("QUIT",false);
}
catch
{
isSended=false;
}
finally
{
this.disconnect();
//輸出日志文件
if(this.LogPath!=null)
{
FileStream fs=null;
if(File.Exists(this.LogPath))
{
fs=new FileStream(this.LogPath,FileMode.Append,FileAccess.Write);
this.logs=CRLF+CRLF+this.logs;
}
else
fs=new FileStream(this.LogPath,FileMode.Create,FileAccess.Write);
byte[] logPath_b=Encoding.GetEncoding("gb2312").GetBytes(this.logs);
fs.Write(logPath_b,0,logPath_b.Length);
fs.Close();
}
}
return isSended;
}
/// <summary>
/// 異步寫入數據
/// </summary>
/// <param name="result"></param>
private void asyncCallBack(IAsyncResult result)
{
if(result.IsCompleted)
this.sendIsComplete=true;
}
/// <summary>
/// 關閉連接
/// </summary>
private void disconnect()
{
try
{
ns.Close();
tc.Close();
}
catch
{
;
}
}
/// <summary>
/// 設置出現錯誤時的動作
/// </summary>
/// <param name="errorStr"></param>
private void setError(string errorStr)
{
this.errorMessage=errorStr;
logs+=this.errorMessage+CRLF+"【郵件處理動作中止】"+CRLF;
this.disconnect();
throw new ApplicationException("");
}
/// <summary>
///將字符串轉換為base64
/// </summary>
/// <param name="str"></param>
/// <param name="encodingName"></param>
/// <returns></returns>
private string convertBase64String(string str,string encodingName)
{
byte[] str_b=Encoding.GetEncoding(encodingName).GetBytes(str);
return Convert.ToBase64String(str_b,0,str_b.Length);
}
#endregion
}
}