.Net FrameWork 框架下使用System.Net.Mail封装类 发送邮件失败:服务器响应
偶然兴起,想做一个后台监控PLC状态的服务。功能如下:监控到PLC状态值异常后触发邮件推送,状态改变后只推送一次。开始使用的是.net6.0开发框架开发,一切都很顺利,邮件也能正常推送。但由于现场工控机系统不是WIN10 20H2的最新版本,导致系统未安装.Net6.0 Runtime。而我也没有再去安装的打算。我重新使用了.net FrameWork4.7 框架进行开发。开发完成后,我以为能正常运行。但出现了不可预知的错误——服务器响应:5.7.1 Client was not authenticated。下面分别是2个框架下发送邮件的代码:.Net 6.0框架:
点击查看代码public bool Send()
{
try
{
SmtpClient smtp = new SmtpClient(this.Host!, (int)this.Port!) { Credentials = new NetworkCredential(this.User!, this.Password!), EnableSsl = true, UseDefaultCredentials = false };
MailMessage message = new MailMessage(this.SenderAddress!, this.ReciverAddress!, this.Subject, this.Body) { From = new MailAddress(this.SenderAddress!, this.SenderName) };
ServicePointManager.ServerCertificateValidationCallback = delegate (object obj, X509Certificate certificate, X509Chain chain, SslPolicyErrors errors)
{
return true;
};
smtp.Send(message);
return true;
}
catch (Exception)
{
return false;
}
}.Net FrameWork 4.7 框架:
点击查看代码public bool Send()
{
try
{
SmtpClient smtp = new SmtpClient(this.Host!, (int)this.Port!) { Credentials = new NetworkCredential(this.User!, this.Password!), EnableSsl = true, UseDefaultCredentials = false };
MailMessage message = new MailMessage(this.SenderAddress!, this.ReciverAddress!, this.Subject, this.Body) { From = new MailAddress(this.SenderAddress!, this.SenderName) };
ServicePointManager.ServerCertificateValidationCallback = delegate (object obj, X509Certificate certificate, X509Chain chain, SslPolicyErrors errors)
{
return true;
};
smtp.Send(message);
return true;
}
catch (Exception)
{
return false;
}
}在不同的开发框架下,使用的代码完全一致。但是在.Net FrameWork 4.7 框架下发送邮件才会出现异常:**5.7.1 Client was not authenticated**,而.Net 6.0 环境下不会。我不得不怀疑是不是微软的封装类System.Net.Mail存在问题。经过断点调试,终于发现了2个环境发送邮件时存在的差异。
.Net 6.0框架下用户传入的凭证(账号密码)SMTP服务器可正常获取到
.Net FrameWork 框架下竟然获取的凭证为空
经过短暂思考,我决定修改下.Net FrameWork框架下的开发代码。如下:
点击查看代码public bool Send()
{
try
{
smtp = new SmtpClient(this.Host, (int)this.Port);
smtp.Credentials = new NetworkCredential(this.User,this.Password);
smtp.EnableSsl = true;
//smtp.UseDefaultCredentials = false;
MailMessage message = new MailMessage(this.SenderAddress, this.ReciverAddress, this.Subject, this.Body) { From = new MailAddress(this.SenderAddress, this.SenderName) };
ServicePointManager.ServerCertificateValidationCallback = delegate (object obj, X509Certificate certificate, X509Chain chain, SslPolicyErrors errors)
{
return true;
};
smtp.Send(message);
return true;
}
catch (Exception)
{
return false;
}
}上述代码中,取消了UseDefaultCredentials属性的使用。运行后,邮件发送正常。那么问题来了,为什么使用了UseDefaultCredentials属性就会导致Credentials 凭证为空呢?看下.Net FrameWork框架System.Net.Mail源码:点击查看代码public bool UseDefaultCredentials
{
get
{
if (!(transport.Credentials is SystemNetworkCredential))
{
return false;
}
return true;
}
set
{
if (InCall)
{
throw new InvalidOperationException(SR.GetString("SmtpInvalidOperationDuringSend"));
}
transport.Credentials = (value ? CredentialCache.DefaultNetworkCredentials : null);
}
}再对比下.net6.0 框架System.Net.Mail源码:点击查看代码 //
// 摘要:
// Gets or sets a System.Boolean value that controls whether the System.Net.CredentialCache.DefaultCredentials
// are sent with requests.
//
// 返回结果:
// true if the default credentials are used; otherwise false. The default value
// is false.
//
// 异常:
// T:System.InvalidOperationException:
// You cannot change the value of this property when an email is being sent.
public bool UseDefaultCredentials
{
get;
set;
}水落石出,微软误我啊!
来源:https://www.cnblogs.com/FlowerFly/Undeclared/17945512
免责声明:由于采集信息均来自互联网,如果侵犯了您的权益,请联系我们【E-Mail:cb@itdo.tech】 我们会及时删除侵权内容,谢谢合作!
页:
[1]