在Office365上使用默认SMTP凭据的例外 - 客户端未经过身份验证,无法在MAIL FROM期间发送匿名邮件

时间:2015-04-15 10:57:04

标签: c# email authentication smtp office365

我使用NLog将日志作为带有自定义邮件目标的电子邮件发送。我从我的office365帐户发送设置为我的web.config(我的主项目)中的默认帐户,如下所示:

  <system.net>
    <mailSettings>
      <smtp deliveryMethod="Network" from="myusername@mydomain.com">
        <network defaultCredentials="false" host="smtp.office365.com" port="587" userName="myusername@mydomain.com" password="mypassword" enableSsl="true" />
      </smtp>
    </mailSettings>
  </system.net>

我用我的日志目标(在我的NLog实现包中)覆盖Write方法,如下所示:

    protected override void Write(LogEventInfo logEvent) 
    { 
        try
        {
            using (var mail = new MailMessage())
            {
                this.SetupMailMessage(mail, logEvent, this.Layout.Render(logEvent));

                using (SmtpClient smtpClient = new SmtpClient())
                {
                    smtpClient.UseDefaultCredentials = true;
                    smtpClient.Send(mail);
                }
            }
        }
        catch (Exception exception)
        {
            throw new NLogRuntimeException("An error occurred when sending a log mail message.", exception);
        }
    }

当系统尝试从此帐户发送邮件时,会抛出以下System.Net.Mail.SmtpException

SMTP服务器需要安全连接,或者客户端未经过身份验证。服务器响应是:5.7.57 SMTP;客户端未通过身份验证,无法在MAIL FROM

期间发送匿名邮件

我已经检查了凭据四倍,它们是正确的。有谁知道还有什么可能导致这个例外?

UPDATE:事实证明,CredentialCache.DefaultCredentials属性中充满了空字符串。然而,当我使用下面的代码手动提取设置时,我可以从web.config获取设置。

SmtpSection settings = (SmtpSection)ConfigurationManager.GetSection("system.net/mailSettings/smtp");
smtpClient.Credentials = new NetworkCredential(settings.Network.UserName, settings.Network.Password);
smtpClient.Host = settings.Network.Host;
smtpClient.Port = settings.Network.Port;
smtpClient.EnableSsl = settings.Network.EnableSsl;

var creds = CredentialCache.DefaultCredentials;  // Is empty

我可以将其用作解决方法。但是给出了什么?为什么默认凭据为空?

4 个答案:

答案 0 :(得分:10)

感谢这篇文章,我能够解决我们的问题。我们通过Rackspace的混合设置将邮箱迁移到O365。用于发送的邮箱以前不是Exchange帐户,但在迁移后成为一个帐户。

mySmtpClient = New SmtpClient("smtp.office365.com")
mySmtpClient.Port = 587
mySmtpClient.EnableSsl = True
mySmtpClient.Credentials = New System.Net.NetworkCredential("email@domain.com", "password", "domain.com")
mySmtpClient.Send(Msg)

以前的设置不要求我们提供端口或启用ssl,甚至不要将域放在凭证参数中。希望这对那些必须使用VB脚本的人们有所帮助,这些脚本通过SMTP与Office 365自动发送电子邮件。

答案 1 :(得分:7)

虽然我在答案更新中提到的解决方法确实有效,但我对手动获取这些值感到不满意。我的解决方案是删除行

smtpClient.UseDefaultCredentials = true;

来自我发布的原始代码。事实证明,使用我在web.config中设置的默认凭据初始化smtpClient,上面删除的行用CredentialCache.DefaultCredentials中的空字符串覆盖它们。我仍然不知道为什么CredentialCache.DefaultCredentials为空或者应该从web.config填充这个,但这是我的问题的根源。

如果有人对此有任何进一步的了解,请发布更好的答案!

答案 2 :(得分:1)

在第三方托管服务提供商上进行此集成时,您还需要提供域名。

SmtpClient client = new SmtpClient(SMTPSettings.Url, SMTPSettings.Port);
client.Credentials = new System.Net.NetworkCredential(SMTPSettings.UserName, SMTPSettings.Password, SMTPSettings.Domain);

答案 3 :(得分:0)

我有同样的问题,它对我有用的是使用System.Security.SecureString来输入密码而不是字符串,例如:

        System.Security.SecureString psw = new System.Security.SecureString();

        string PasswordGmail = "XXXXXX";

        foreach (char item in PasswordGmail.ToCharArray())
        {
            psw.AppendChar(item);
        }

        client.Credentials = new System.Net.NetworkCredential("XXXX@xxx.com", psw);