2

我目前在 MVC3 中提供一个表单供用户填写一些字段并附加文件。提交时,我将发布的信息(带有附件)发送两次......一次作为收据发送给海报,第二次发送给另一封目标电子邮件。

我遇到的问题是第一封电子邮件使用正确的附件成功发送。第二封电子邮件会发送一个大小为 0 的附件。似乎从文件上传制作附件对象后,我无法再次重复使用它。使用调试器我可以看到文件上传对象仍在内存中,但其 ContentLength 变为 0。

所以在下面的例子中,如果我简化我的代码如下:

public static void SendDummyEmail1()
{
    using (var mailMessage = new MailMessage("from@email.com", "to@email.com"))
    {
        mailMessage.Subject = "Email Subject"
        mailMessage.Body = Razor.Parse(template, (dynamic)dynamicTokens);
        mailMessage.IsBodyHtml = true;

        if (_fileUpload != null && _fileUpload.ContentLength > 0)
        {
            var attachment = new Attachment(_fileUpload.InputStream, _fileUpload.FileName, MediaTypeNames.Application.Octet);
            attachment.ContentDisposition.FileName = Path.GetFileName(_fileUpload.FileName);
            mailMessage.Attachments.Add(attachment);
        }

        SendMail(mailMessage);
    }
}

public static void SendMail(MailMessage message)
{
    var client = new SmtpClient
    {
        Host = ConfigurationManager.AppSettings[SmtpHostname],
        Port = Convert.ToInt32(ConfigurationManager.AppSettings[SmtpPortNumber]),
        UseDefaultCredentials = true,
        Credentials = CredentialCache.DefaultNetworkCredentials,
        DeliveryMethod = SmtpDeliveryMethod.Network,
        EnableSsl = true,  
    };

    // Work around remote certificate validation
    // Ref: http://stackoverflow.com/questions/777607/the-remote-certificate-is-invalid-according-to-the-validation-procedure-using
    ServicePointManager.ServerCertificateValidationCallback = delegate { return true; };

    client.Send(message);
}

如果我连续两次调用上述方法,第一封电子邮件将成功通过。第二封电子邮件将通过,但没有附件,因为 ContentLength 变为 0。

SendDummyEmail1();
SendDummyEmail1();
4

3 回答 3

3

我想_fileUpload.InputStream.CanSeek在你的情况下等于false,这意味着你不能再次结束它 ( Position = 0) 并从中读取。尝试先将上传的文件流复制到内存流中,然后使用它而不是初始流,如下所示:

MemoryStream ms = new MemoryStream();
_fileUpload.InputStream.CopyTo(ms);
byte[] data = ms.ToArray();
SendDummyEmail1(data);
SendDummyEmail1(data);

public static void SendDummyEmail1(byte[] fileContent)
{
 ...
 var attachment = new Attachment(new MemoryStream(fileContent), ...
}
于 2013-02-28T10:42:07.617 回答
3

使用内存流作为实例变量而不是 _fileUpload。然后将其克隆到每封电子邮件的新蒸汽中(因此不要有多个消费者)。请记住在写入流后重置流,以便从头开始读取附件

于 2013-02-28T10:56:14.590 回答
0

将上传的文件存储在服务器上并使用附件中的文件路径。

var attachment = new Attachment(filePath, MediaTypeNames.Application.Octet);

发送电子邮件后,如果需要删除上传的文件。

于 2013-02-28T10:10:40.617 回答