通用类 SmtpHelper 邮件附件
public enum MailFormat { Text, HTML }; public enum MailPriority { Low = 1, Normal = 3, High = 5 }; #region Class mailAttachments public class MailAttachments { private const int MaxAttachmentNum = 10; private IList _Attachments; public MailAttachments() { _Attachments = new ArrayList(); } public string this [ int index] { get { return ( string )_Attachments[index]; } } /// <summary> /// 添加邮件附件 /// </summary> /// <param name="FilePath">附件的绝对路径</param> public void Add( params string [] filePath) { if (filePath == null ) { throw ( new ArgumentNullException( "非法的附件" )); } else { for ( int i = 0; i < filePath.Length; i++) { Add(filePath[i]); } } } /// <summary> /// 添加一个附件,当指定的附件不存在时,忽略该附件,不产生异常。 /// </summary> /// <param name="filePath">附件的绝对路径</param> public void Add( string filePath) { //当附件存在时才加入,否则忽略 if (System.IO.File.Exists(filePath)) { if (_Attachments.Count < MaxAttachmentNum) { _Attachments.Add(filePath); } } } public void Clear() //清除所有附件 { _Attachments.Clear(); } public int Count //获取附件个数 { get { return _Attachments.Count; } } } #endregion//end Class mailAttachments #region Class MailMessage /// <summary> /// MailMessage 表示SMTP要发送的一封邮件的消息。 /// </summary> public class MailMessage { private const int MaxRecipientNum = 10; public MailMessage() { _Recipients = new ArrayList(); //收件人列表 _Attachments = new MailAttachments(); //附件 _BodyFormat = MailFormat.Text; //缺省的邮件格式为Text _Priority = MailPriority.Normal; _Charset = "UTF-8" ; } /// <summary> /// 设定语言代码,默认设定为GB2312,如不需要可设置为"" /// </summary> public string Charset { get { return _Charset; } set { _Charset = value; } } public string From { get { return _From; } set { _From = value; } } public string FromName { get { return _FromName; } set { _FromName = value; } } public string Body { get { return _Body; } set { _Body = value; } } public string Subject { get { return _Subject; } set { _Subject = value; } } public MailAttachments Attachments { get { return _Attachments; } set { _Attachments = value; } } public MailPriority Priority { get { return _Priority; } set { _Priority = value; } } public IList Recipients { get { return _Recipients; } } /// <summary> /// 增加一个收件人地址 /// </summary> /// <param name="recipient">收件人的Email地址</param> public void AddRecipients( string recipient) { //先检查邮件地址是否符合规范 if (_Recipients.Count < MaxRecipientNum) { _Recipients.Add(recipient); //增加到收件人列表 } } public void AddRecipients( params string [] recipient) { if (recipient == null ) { throw ( new ArgumentException( "收件人不能为空." )); } else { for ( int i = 0; i < recipient.Length; i++) { AddRecipients(recipient[i]); } } } public MailFormat BodyFormat { set { _BodyFormat = value; } get { return _BodyFormat; } } private string _From; //发件人地址 private string _FromName; //发件人姓名 private IList _Recipients; //收件人 private MailAttachments _Attachments; //附件 private string _Body; //内容 private string _Subject; //主题 private MailFormat _BodyFormat; //邮件格式 private string _Charset = "GB2312" ; //字符编码格式 private MailPriority _Priority; //邮件优先级 } #endregion #region Class SmtpMail public class SmtpServerHelper { private string CRLF = "\r\n" ; //回车换行 /// <summary> /// 错误消息反馈 /// </summary> private string errmsg; /// <summary> /// TcpClient对象,用于连接服务器 /// </summary> private TcpClient tcpClient; /// <summary> /// NetworkStream对象 /// </summary> private NetworkStream networkStream; /// <summary> /// 服务器交互记录 /// </summary> private string logs = "" ; /// <summary> /// SMTP错误代码哈希表 /// </summary> private Hashtable ErrCodeHT = new Hashtable(); /// <summary> /// SMTP正确代码哈希表 /// </summary> private Hashtable RightCodeHT = new Hashtable(); public SmtpServerHelper() { SMTPCodeAdd(); //初始化SMTPCode } ~SmtpServerHelper() { networkStream.Close(); tcpClient.Close(); } /// <summary> /// 将字符串编码为Base64字符串 /// </summary> /// <param name="str">要编码的字符串</param> private string Base64Encode( string str) { byte [] barray; barray = Encoding.Default.GetBytes(str); return Convert.ToBase64String(barray); } /// <summary> /// 将Base64字符串解码为普通字符串 /// </summary> /// <param name="str">要解码的字符串</param> private string Base64Decode( string str) { byte [] barray; barray = Convert.FromBase64String(str); return Encoding.Default.GetString(barray); } /// <summary> /// 得到上传附件的文件流 /// </summary> /// <param name="FilePath">附件的绝对路径</param> private string GetStream( string FilePath) { //建立文件流对象 System.IO.FileStream FileStr = new System.IO.FileStream(FilePath, System.IO.FileMode.Open); byte [] by = new byte [System.Convert.ToInt32(FileStr.Length)]; FileStr.Read( by , 0, by .Length); FileStr.Close(); return (System.Convert.ToBase64String( by )); } /// <summary> /// SMTP回应代码哈希表 /// </summary> private void SMTPCodeAdd() { //[RFC 821 4.2.1.] /* 4.2.2. NUMERIC ORDER LIST OF REPLY CODES 211 System status, or system help reply 214 Help message [Information on how to use the receiver or the meaning of a particular non-standard command; this reply is useful only to the human user] 220 <domain> Service ready 221 <domain> Service closing transmission channel 250 Requested mail action okay, completed 251 User not local; will forward to <forward-path> 354 Start mail input; end with <CRLF>.<CRLF> 421 <domain> Service not available, closing transmission channel [This may be a reply to any command if the service knows it must shut down] 450 Requested mail action not taken: mailbox unavailable [E.g., mailbox busy] 451 Requested action aborted: local error in processing 452 Requested action not taken: insufficient system storage 500 Syntax error, command unrecognized [This may include errors such as command line too long] 501 Syntax error in parameters or arguments 502 Command not implemented 503 Bad sequence of commands 504 Command parameter not implemented 550 Requested action not taken: mailbox unavailable [E.g., mailbox not found, no access] 551 User not local; please try <forward-path> 552 Requested mail action aborted: exceeded storage allocation 553 Requested action not taken: mailbox name not allowed [E.g., mailbox syntax incorrect] 554 Transaction failed */ ErrCodeHT.Add( "421" , "服务未就绪,关闭传输信道" ); ErrCodeHT.Add( "432" , "需要一个密码转换" ); ErrCodeHT.Add( "450" , "要求的邮件操作未完成,邮箱不可用(例如,邮箱忙)" ); ErrCodeHT.Add( "451" , "放弃要求的操作;处理过程中出错" ); ErrCodeHT.Add( "452" , "系统存储不足,要求的操作未执行" ); ErrCodeHT.Add( "454" , "临时认证失败" ); ErrCodeHT.Add( "500" , "邮箱地址错误" ); ErrCodeHT.Add( "501" , "参数格式错误" ); ErrCodeHT.Add( "502" , "命令不可实现" ); ErrCodeHT.Add( "503" , "服务器需要SMTP验证" ); ErrCodeHT.Add( "504" , "命令参数不可实现" ); ErrCodeHT.Add( "530" , "需要认证" ); ErrCodeHT.Add( "534" , "认证机制过于简单" ); ErrCodeHT.Add( "538" , "当前请求的认证机制需要加密" ); ErrCodeHT.Add( "550" , "要求的邮件操作未完成,邮箱不可用(例如,邮箱未找到,或不可访问)" ); ErrCodeHT.Add( "551" , "用户非本地,请尝试<forward-path>" ); ErrCodeHT.Add( "552" , "过量的存储分配,要求的操作未执行" ); ErrCodeHT.Add( "553" , "邮箱名不可用,要求的操作未执行(例如邮箱格式错误)" ); ErrCodeHT.Add( "554" , "传输失败" ); /* 211 System status, or system help reply 214 Help message [Information on how to use the receiver or the meaning of a particular non-standard command; this reply is useful only to the human user] 220 <domain> Service ready 221 <domain> Service closing transmission channel 250 Requested mail action okay, completed 251 User not local; will forward to <forward-path> 354 Start mail input; end with <CRLF>.<CRLF> */ RightCodeHT.Add( "220" , "服务就绪" ); RightCodeHT.Add( "221" , "服务关闭传输信道" ); RightCodeHT.Add( "235" , "验证成功" ); RightCodeHT.Add( "250" , "要求的邮件操作完成" ); RightCodeHT.Add( "251" , "非本地用户,将转发向<forward-path>" ); RightCodeHT.Add( "334" , "服务器响应验证Base64字符串" ); RightCodeHT.Add( "354" , "开始邮件输入,以<CRLF>.<CRLF>结束" ); } /// <summary> /// 发送SMTP命令 /// </summary> private bool SendCommand( string str) { byte [] WriteBuffer; if (str == null || str.Trim() == String.Empty) { return true ; } logs += str; WriteBuffer = Encoding.Default.GetBytes(str); try { networkStream.Write(WriteBuffer, 0, WriteBuffer.Length); } catch { errmsg = "网络连接错误" ; return false ; } return true ; } /// <summary> /// 接收SMTP服务器回应 /// </summary> private string RecvResponse() { int StreamSize; string Returnvalue = String.Empty; byte [] ReadBuffer = new byte [1024]; try { StreamSize = networkStream.Read(ReadBuffer, 0, ReadBuffer.Length); } catch { errmsg = "网络连接错误" ; return "false" ; } if (StreamSize == 0) { return Returnvalue; } else { Returnvalue = Encoding.Default.GetString(ReadBuffer).Substring(0, StreamSize); logs += Returnvalue + this .CRLF; return Returnvalue; } } /// <summary> /// 与服务器交互,发送一条命令并接收回应。 /// </summary> /// <param name="str">一个要发送的命令</param> /// <param name="errstr">如果错误,要反馈的信息</param> private bool Dialog( string str, string errstr) { if (str == null || str.Trim() == string .Empty) { return true ; } if (SendCommand(str)) { string RR = RecvResponse(); if (RR == "false" ) { return false ; } //检查返回的代码,根据[RFC 821]返回代码为3位数字代码如220 string RRCode = RR.Substring(0, 3); if (RightCodeHT[RRCode] != null ) { return true ; } else { if (ErrCodeHT[RRCode] != null ) { errmsg += (RRCode + ErrCodeHT[RRCode].ToString()); errmsg += CRLF; } else { errmsg += RR; } errmsg += errstr; return false ; } } else { return false ; } } /// <summary> /// 与服务器交互,发送一组命令并接收回应。 /// </summary> private bool Dialog( string [] str, string errstr) { for ( int i = 0; i < str.Length; i++) { if (!Dialog(str[i], "" )) { errmsg += CRLF; errmsg += errstr; return false ; } } return true ; } //连接服务器 private bool Connect( string smtpServer, int port) { //创建Tcp连接 try { tcpClient = new TcpClient(smtpServer, port); } catch (Exception e) { errmsg = e.ToString(); return false ; } networkStream = tcpClient.GetStream(); //验证网络连接是否正确 if (RightCodeHT[RecvResponse().Substring(0, 3)] == null ) { errmsg = "网络连接失败" ; return false ; } return true ; } private string GetPriorityString(MailPriority mailPriority) { string priority = "Normal" ; if (mailPriority == MailPriority.Low) { priority = "Low" ; } else if (mailPriority == MailPriority.High) { priority = "High" ; } return priority; } /// <summary> /// 发送电子邮件,SMTP服务器不需要身份验证 /// </summary> /// <param name="smtpServer"></param> /// <param name="port"></param> /// <param name="mailMessage"></param> /// <returns></returns> public bool SendEmail( string smtpServer, int port, MailMessage mailMessage) { return SendEmail(smtpServer, port, false , "" , "" , mailMessage); } /// <summary> /// 发送电子邮件,SMTP服务器需要身份验证 /// </summary> /// <param name="smtpServer"></param> /// <param name="port"></param> /// <param name="username"></param> /// <param name="password"></param> /// <param name="mailMessage"></param> /// <returns></returns> public bool SendEmail( string smtpServer, int port, string username, string password, MailMessage mailMessage) { return SendEmail(smtpServer, port, false , username, password, mailMessage); } private bool SendEmail( string smtpServer, int port, bool ESmtp, string username, string password, MailMessage mailMessage) { if (Connect(smtpServer, port) == false ) //测试连接服务器是否成功 return false ; string priority = GetPriorityString(mailMessage.Priority); bool Html = (mailMessage.BodyFormat == MailFormat.HTML); string [] SendBuffer; string SendBufferstr; //进行SMTP验证,现在大部分SMTP服务器都要认证 if (ESmtp) { SendBuffer = new String[4]; SendBuffer[0] = "EHLO " + smtpServer + CRLF; SendBuffer[1] = "AUTH LOGIN" + CRLF; SendBuffer[2] = Base64Encode(username) + CRLF; SendBuffer[3] = Base64Encode(password) + CRLF; if (!Dialog(SendBuffer, "SMTP服务器验证失败,请核对用户名和密码。" )) return false ; } else { //不需要身份认证 SendBufferstr = "HELO " + smtpServer + CRLF; if (!Dialog(SendBufferstr, "" )) return false ; } //发件人地址 SendBufferstr = "MAIL FROM:<" + mailMessage.From + ">" + CRLF; if (!Dialog(SendBufferstr, "发件人地址错误,或不能为空" )) return false ; //收件人地址 SendBuffer = new string [mailMessage.Recipients.Count]; for ( int i = 0; i < mailMessage.Recipients.Count; i++) { SendBuffer[i] = "RCPT TO:<" + ( string )mailMessage.Recipients[i] + ">" + CRLF; } if (!Dialog(SendBuffer, "收件人地址有误" )) return false ; /* SendBuffer=new string[10]; for(int i=0;i<RecipientBCC.Count;i++) { SendBuffer[i]="RCPT TO:<" + RecipientBCC[i].ToString() +">" + CRLF; } if(!Dialog(SendBuffer,"密件收件人地址有误")) return false; */ SendBufferstr = "DATA" + CRLF; if (!Dialog(SendBufferstr, "" )) return false ; //发件人姓名 SendBufferstr = "From:" + mailMessage.FromName + "<" + mailMessage.From + ">" + CRLF; //if(ReplyTo.Trim()!="") //{ // SendBufferstr+="Reply-To: " + ReplyTo + CRLF; //} //SendBufferstr+="To:" + ToName + "<" + Recipient[0] +">" +CRLF; //至少要有一个收件人 if (mailMessage.Recipients.Count == 0) { return false ; } else { SendBufferstr += "To:=?" + mailMessage.Charset.ToUpper() + "?B?" + Base64Encode(( string )mailMessage.Recipients[0]) + "?=" + "<" + ( string )mailMessage.Recipients[0] + ">" + CRLF; } //SendBufferstr+="CC:"; //for(int i=0;i<Recipient.Count;i++) //{ // SendBufferstr+=Recipient[i].ToString() + "<" + Recipient[i].ToString() +">,"; //} //SendBufferstr+=CRLF; SendBufferstr += ((mailMessage.Subject == String.Empty || mailMessage.Subject == null ) ? "Subject:" : ((mailMessage.Charset == "" ) ? ( "Subject:" + mailMessage.Subject) : ( "Subject:" + "=?" + mailMessage.Charset.ToUpper() + "?B?" + Base64Encode(mailMessage.Subject) + "?=" ))) + CRLF; SendBufferstr += "X-Priority:" + priority + CRLF; SendBufferstr += "X-MSMail-Priority:" + priority + CRLF; SendBufferstr += "Importance:" + priority + CRLF; SendBufferstr += "X-Mailer: Lion.Web.Mail.SmtpMail Pubclass [cn]" + CRLF; SendBufferstr += "MIME-Version: 1.0" + CRLF; if (mailMessage.Attachments.Count != 0) { SendBufferstr += "Content-Type: multipart/mixed;" + CRLF; SendBufferstr += " boundary=\"=====" + (Html ? "001_Dragon520636771063_" : "001_Dragon303406132050_" ) + "=====\"" + CRLF + CRLF; } if (Html) { if (mailMessage.Attachments.Count == 0) { SendBufferstr += "Content-Type: multipart/alternative;" + CRLF; //内容格式和分隔符 SendBufferstr += " boundary=\"=====003_Dragon520636771063_=====\"" + CRLF + CRLF; SendBufferstr += "This is a multi-part message in MIME format." + CRLF + CRLF; } else { SendBufferstr += "This is a multi-part message in MIME format." + CRLF + CRLF; SendBufferstr += "--=====001_Dragon520636771063_=====" + CRLF; SendBufferstr += "Content-Type: multipart/alternative;" + CRLF; //内容格式和分隔符 SendBufferstr += " boundary=\"=====003_Dragon520636771063_=====\"" + CRLF + CRLF; } SendBufferstr += "--=====003_Dragon520636771063_=====" + CRLF; SendBufferstr += "Content-Type: text/plain;" + CRLF; SendBufferstr += ((mailMessage.Charset == "" ) ? ( " charset=\"iso-8859-1\"" ) : ( " charset=\"" + mailMessage.Charset.ToLower() + "\"" )) + CRLF; SendBufferstr += "Content-Transfer-Encoding: base64" + CRLF + CRLF; SendBufferstr += Base64Encode( "邮件内容为HTML格式,请选择HTML方式查看" ) + CRLF + CRLF; SendBufferstr += "--=====003_Dragon520636771063_=====" + CRLF; SendBufferstr += "Content-Type: text/html;" + CRLF; SendBufferstr += ((mailMessage.Charset == "" ) ? ( " charset=\"iso-8859-1\"" ) : ( " charset=\"" + mailMessage.Charset.ToLower() + "\"" )) + CRLF; SendBufferstr += "Content-Transfer-Encoding: base64" + CRLF + CRLF; SendBufferstr += Base64Encode(mailMessage.Body) + CRLF + CRLF; SendBufferstr += "--=====003_Dragon520636771063_=====--" + CRLF; } else { if (mailMessage.Attachments.Count != 0) { SendBufferstr += "--=====001_Dragon303406132050_=====" + CRLF; } SendBufferstr += "Content-Type: text/plain;" + CRLF; SendBufferstr += ((mailMessage.Charset == "" ) ? ( " charset=\"iso-8859-1\"" ) : ( " charset=\"" + mailMessage.Charset.ToLower() + "\"" )) + CRLF; SendBufferstr += "Content-Transfer-Encoding: base64" + CRLF + CRLF; SendBufferstr += Base64Encode(mailMessage.Body) + CRLF; } //SendBufferstr += "Content-Transfer-Encoding: base64"+CRLF; if (mailMessage.Attachments.Count != 0) { for ( int i = 0; i < mailMessage.Attachments.Count; i++) { string filepath = ( string )mailMessage.Attachments[i]; SendBufferstr += "--=====" + (Html ? "001_Dragon520636771063_" : "001_Dragon303406132050_" ) + "=====" + CRLF; //SendBufferstr += "Content-Type: application/octet-stream"+CRLF; SendBufferstr += "Content-Type: text/plain;" + CRLF; SendBufferstr += " name=\"=?" + mailMessage.Charset.ToUpper() + "?B?" + Base64Encode(filepath.Substring(filepath.LastIndexOf( "\\" ) + 1)) + "?=\"" + CRLF; SendBufferstr += "Content-Transfer-Encoding: base64" + CRLF; SendBufferstr += "Content-Disposition: attachment;" + CRLF; SendBufferstr += " filename=\"=?" + mailMessage.Charset.ToUpper() + "?B?" + Base64Encode(filepath.Substring(filepath.LastIndexOf( "\\" ) + 1)) + "?=\"" + CRLF + CRLF; SendBufferstr += GetStream(filepath) + CRLF + CRLF; } SendBufferstr += "--=====" + (Html ? "001_Dragon520636771063_" : "001_Dragon303406132050_" ) + "=====--" + CRLF + CRLF; } SendBufferstr += CRLF + "." + CRLF; //内容结束 if (!Dialog(SendBufferstr, "错误信件信息" )) return false ; SendBufferstr = "QUIT" + CRLF; if (!Dialog(SendBufferstr, "断开连接时错误" )) return false ; networkStream.Close(); tcpClient.Close(); return true ; } } public class SmtpMail { private static string _SmtpServer; /// <summary> /// 格式:SmtpAccount:Password@SmtpServerAddress<br> /// 或者:SmtpServerAddress<br> /// <code> /// SmtpMail.SmtpServer="user:12345678@smtp.126.com"; /// //或者: /// SmtpMail.SmtpServer="smtp.126.com"; /// 或者: /// SmtpMail.SmtpServer=SmtpServerHelper.GetSmtpServer("user","12345678","smtp.126.com"); /// </code> /// </summary> public static string SmtpServer { set { _SmtpServer = value; } get { return _SmtpServer; } } public static bool Send(MailMessage mailMessage, string username, string password) { SmtpServerHelper helper = new SmtpServerHelper(); return helper.SendEmail(_SmtpServer, 25, username, password, mailMessage); } } #endregion |
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 深入理解 Mybatis 分库分表执行原理
· 如何打造一个高并发系统?
· .NET Core GC压缩(compact_phase)底层原理浅谈
· 现代计算机视觉入门之:什么是图片特征编码
· .NET 9 new features-C#13新的锁类型和语义
· Spring AI + Ollama 实现 deepseek-r1 的API服务和调用
· 《HelloGitHub》第 106 期
· 数据库服务器 SQL Server 版本升级公告
· 深入理解Mybatis分库分表执行原理
· 使用 Dify + LLM 构建精确任务处理应用