16

Is there any easy way to convert a System.Net.Mail.MailMessage object to the raw mail message text, like when you open a eml file in notepad.

5 Answers 5

19

Here's the same solution, but as an extension method to MailMessage.

Some of the reflection overhead is minimized by grabbing the ConstructorInfo and MethodInfo members once in the static context.

/// <summary> /// Uses reflection to get the raw content out of a MailMessage. /// </summary> public static class MailMessageExtensions { private static readonly BindingFlags Flags = BindingFlags.Instance | BindingFlags.NonPublic; private static readonly Type MailWriter = typeof(SmtpClient).Assembly.GetType("System.Net.Mail.MailWriter"); private static readonly ConstructorInfo MailWriterConstructor = MailWriter.GetConstructor(Flags, null, new[] { typeof(Stream) }, null); private static readonly MethodInfo CloseMethod = MailWriter.GetMethod("Close", Flags); private static readonly MethodInfo SendMethod = typeof(MailMessage).GetMethod("Send", Flags); /// <summary> /// A little hack to determine the number of parameters that we /// need to pass to the SaveMethod. /// </summary> private static readonly bool IsRunningInDotNetFourPointFive = SendMethod.GetParameters().Length == 3; /// <summary> /// The raw contents of this MailMessage as a MemoryStream. /// </summary> /// <param name="self">The caller.</param> /// <returns>A MemoryStream with the raw contents of this MailMessage.</returns> public static MemoryStream RawMessage(this MailMessage self) { var result = new MemoryStream(); var mailWriter = MailWriterConstructor.Invoke(new object[] { result }); SendMethod.Invoke(self, Flags, null, IsRunningInDotNetFourPointFive ? new[] { mailWriter, true, true } : new[] { mailWriter, true }, null); result = new MemoryStream(result.ToArray()); CloseMethod.Invoke(mailWriter, Flags, null, new object[] { }, null); return result; } } 

To grab the underlying MemoryStream:

var email = new MailMessage(); using (var m = email.RawMessage()) { // do something with the raw message } 
Sign up to request clarification or add additional context in comments.

8 Comments

Does this work (with the additional parameter for Send) in .NET 4.5 on your PC? I had to copy the content of result to another MemoryStream before calling CloseMethod, because the current implementation will actually close the underlying stream.
This works for .NET 4.0 -- I haven't tried it in 4.5. I'd be happy to include your update in my answer (and give you credit, of course).
Yes, it should work like that, but I also added a check for .NET 4.5. I could update the answer with what works, if that's okay with you?
FYI, I noticed that the Send method actually strips out the BCC since it's prepping it for sending. If you are just going to pass this to Amazon SES or some other type of service, that isn't what you want. I ended up just manually creating the MIME string instead.
@allonhadaya, it seems like the answer does not return the MIME encoded-word syntax for non-ASCII characters?
|
9

I've implemented logic in MimeKit to allow you to cast a System.Net.Mail.MailMessage into a MimeKit.MimeMessage. Once you do that, you can simply write the message to a stream:

var message = (MimeMessage) CreateSystemNetMailMessage (); using (var stream = File.Create ("C:\\message.eml")) message.WriteTo (stream); 

This does not require reflecting into internal methods which means that it isn't dependent on the runtime, making it far more portable that the other answers given so far.

5 Comments

I noticed that the reflection method does not get my SubjectEncoding (UTF8) for sending via Amazon SES right. Mimekit does. Thanks!
Awesome! Glad MimeKit has been helpful!
Looks like you're just mapping by hand, so to speak, right? Not that that's horribly bad or a deal breaker, but it's theoretically lossy if I'm not missing something. (??) (Trying to adapt my code to MonoMac now, and was about to give up & do a mapping-by-hand myself.)
Yes, it's just mapping by hand but it should be complete. Obviously mapping is not ideal as a long-term solution and so I would recommend using MimeKit to construct the message as the long-term solution, but the mapping method is useful if you are incrementally porting your S.N.Mail code in the short-term... if that makes sense.
Oh, and if you need SMTP, I have a library called MailKit which adds SMTP, POP3 and IMAP support on top of MimeKit.
2

The code I've seen to do this relies on reflection. I adapted the samples found online to create this method:

 private static MemoryStream ConvertMailMessageToMemoryStream(MailMessage message) { BindingFlags flags = BindingFlags.Instance | BindingFlags.NonPublic; Assembly assembly = typeof(SmtpClient).Assembly; MemoryStream stream = new MemoryStream(); Type mailWriterType = assembly.GetType("System.Net.Mail.MailWriter"); ConstructorInfo mailWriterContructor = mailWriterType.GetConstructor(flags, null, new[] { typeof(Stream) }, null); object mailWriter = mailWriterContructor.Invoke(new object[] { stream }); MethodInfo sendMethod = typeof(MailMessage).GetMethod("Send", flags); sendMethod.Invoke(message, flags, null, new[] { mailWriter, true }, null); MethodInfo closeMethod = mailWriter.GetType().GetMethod("Close", flags); closeMethod.Invoke(mailWriter, flags, null, new object[] { }, null); return stream; } 

You can then convert the MemoryStream to a string or whatever you need.

Update: A method signature has changed in .NET 4.5, which breaks the above: Getting System.Net.Mail.MailMessage as a MemoryStream in .NET 4.5 beta

Comments

0

Fixing @allonhadaya's answer for .NET 8:

/// <summary> /// Uses reflection to get the raw content out of a MailMessage. /// </summary> public static class MailMessageExtensions { private static readonly BindingFlags Flags = BindingFlags.Instance | BindingFlags.NonPublic; private static readonly Type? MailWriter = typeof(SmtpClient).Assembly.GetType("System.Net.Mail.MailWriter"); private static readonly ConstructorInfo? MailWriterConstructor = MailWriter?.GetConstructor(Flags, null, new[] { typeof(Stream), typeof(bool) }, null); private static readonly MethodInfo? CloseMethod = MailWriter?.GetMethod("Close", Flags); private static readonly MethodInfo? SendMethod = typeof(MailMessage).GetMethod("Send", Flags); /// <summary> /// A little hack to determine the number of parameters that we /// need to pass to the SaveMethod. /// </summary> private static readonly bool IsRunningInDotNetFourPointFive = SendMethod?.GetParameters().Length == 3; /// <summary> /// The raw contents of this MailMessage as a MemoryStream. /// </summary> /// <param name="self">The caller.</param> /// <returns>A MemoryStream with the raw contents of this MailMessage.</returns> public static MemoryStream RawMessage(this MailMessage self) { var result = new MemoryStream(); var mailWriter = MailWriterConstructor?.Invoke([result, false]); _ = mailWriter ?? throw new ApplicationException("Could not create a new MailWriter"); _ = SendMethod ?? throw new ApplicationException("Could not find the SendMethod"); SendMethod?.Invoke(self, Flags, null, IsRunningInDotNetFourPointFive ? new[] { mailWriter, true, true } : new[] { mailWriter, true }, null); result = new MemoryStream(result.ToArray()); _ = CloseMethod ?? throw new ApplicationException("Could not find the CloseMethod"); CloseMethod?.Invoke(mailWriter, Flags, null, [], null); return result; } } 

Changed is that the internal MailWriter constructor was changed to take one more bool encodeForTransport parameter (see https://stackoverflow.com/a/73398685/2003763).

Comments

-1
byte[] allBytes = new byte[attachment.ContentStream.Length]; int bytesRead = attachment.ContentStream.Read(allBytes, 0, (int)attachment.ContentStream.Length); Encoding encoding = Encoding.UTF8; String contenidoCorreo = encoding.GetString(allBytes); 

1 Comment

I'm not sure this answer really adds anything without explanation.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.