我正在使用SmtpClient
库通过以下方式发送电子邮件:
SmtpClient client = new SmtpClient();
client.Host = "hostname";
client.Port = 465;
client.DeliveryMethod = SmtpDeliveryMethod.Network;
client.UseDefaultCredentials = false;
client.EnableSsl = true;
client.Credentials = new NetworkCredential("User", "Pass);
client.Send("from@hostname", "to@hostname", "Subject", "Body");
该代码在我的测试环境中可以正常工作,但是当我使用生产SMTP服务器时,该代码会失败,并带有
SmtpException
“发送邮件失败”。带有内部IOException
“无法从传输连接读取数据:net_io_connectionclosed”。我已经确认防火墙不是问题。客户端和服务器之间的端口打开得很好。我不确定还有什么可能引发此错误。
最佳答案
编辑: super Redux版本
请尝试使用端口587而不是465。在技术上,不建议使用端口465。
一堆包嗅探后,我想通了。首先,这是一个简短的答案:
.NET SmtpClient
仅支持通过STARTTLS进行加密。如果设置了EnableSsl
标志,则服务器必须使用STARTTLS响应EHLO,否则它将引发异常。有关更多详细信息,请参见MSDN文档。
其次,为那些在将来偶然发现此问题的人提供了一个简短的SMTP历史类(class):
过去,当服务还希望提供加密时,会为其分配一个不同的端口号,并在该端口号上立即启动SSL连接。随着时间的流逝,他们意识到浪费两个端口号用于一项服务是很愚蠢的,他们设计了一种服务方式,允许使用STARTTLS在同一端口上进行纯文本和加密。通信将开始使用纯文本,然后使用STARTTLS命令升级到加密连接。 STARTTLS成为SMTP加密的标准。不幸的是,由于在实现新标准时总是会发生这种情况,因此与那里的所有客户端和服务器都存在兼容性的大杂烩。
就我而言,我的用户试图将软件连接到强制进行即时SSL连接的服务器,这是.NET中Microsoft不支持的旧方法。
关于c# - SmtpException : Unable to read data from the transport connection: net_io_connectionclosed,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/20228644/