zoukankan      html  css  js  c++  java
  • 通用类 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
    
  • 相关阅读:
    phpqrcode生成带logo的二维码图片及带文字的二维码图片
    php 文件压缩zip扩展
    js常用的正则表达操作
    WebViewJavascriptBridge详细使用(转载)
    html5的FormData对象和input的file属性以及window.URL.createObjectURL( ) 方法(转载)
    js面向对象的实现(example 二)
    PHP二维数组(或任意维数组)转换成一维数组的方法汇总
    Yii2实现自定义独立验证器的方法
    yii2.0配置以pathinfo的形式访问
    安装 AdminLTE和 yii2-admin
  • 原文地址:https://www.cnblogs.com/acyy/p/2662256.html
Copyright © 2011-2022 走看看