Can I turn off the email address validation in System.Net.Mail?
I'm trying to talk to Fax server software using an email. The fax server will accept formatted SMTP mails and covert them to faxes and send them to the fax number defined in the to address. This has been manually tested by sending an email from Outlook via the same server.
Here's my problem - System.Net.Mail is throwing an
System.FormatException: The specified string is not in the form required for an e-mail address.
exception due to the format of the email address that I am trying to send to
Is there any way I can turn off/change this validation because the email address may not be RFC compliant but it will work if the email gets sent
i.e. I want to send to [RFax:User@/FN=0123456789] including the square brackets
You can send to this as an e-mail address in Outlook
Cheers Chris
EDIT
This is a cut-down version of the class I'm using to bypass the validation. There are two ways of doing it - one by overriding the constructor and setting the internal attribute directly, the other using an internal constructor. They have slightly different effects if there are spaces in the email address
using System;
using System.Reflection;
namespace Mail
{
public class UnverifiedEmailAddress : System.Net.Mail.MailAddress
{
/// <summary>
/// Constructor to bypass the validation of MailAddress
/// </summary>
/// <param name="address">Email address to create</param>
public UnverifiedEmailAddress(string address)
: base("a@a")
{
FieldInfo field = typeof(System.Net.Mail.MailAddress).GetField("address", BindingFlags.Instance | BindingFlags.NonPublic);
field.SetValue(this, address);
}
/// <summary>
/// Static method to create an unverifed email address bypassing the address validation
/// </summary>
/// <param name="address">Email address to create</param>
/// <param name="displayName">Display name for email address</param>
/// <returns></returns>
private static System.Net.Mail.MailAddress GetUnverifiedEmailAddress(string address, string displayName)
{
ConstructorInfo cons = typeof(System.Net.Mail.MailAddress).GetConstructor(BindingFlags.Instance | BindingFlags.NonPublic,
null,
new Type[] { typeof(string), typeof(string), typeof(UInt32) },
null);
object obj = cons.Invoke(new object[] { address, displayName, UInt32.MinValue });
System.Net.Mail.MailAddress toAddressObj = (S开发者_StackOverflow中文版ystem.Net.Mail.MailAddress)obj;
return toAddressObj;
}
}
}
No, you cannot turn that validation off.
EDIT:
After looking in to this a little bit it seems as if the following code snippet would be a feasible workaround:
ConstructorInfo ctor = typeof(MailAddress).GetConstructor(
BindingFlags.NonPublic | BindingFlags.Instance, null,
new Type[] { typeof(string), typeof(string), typeof(string) }, null);
MailMessage msg = new MailMessage
{
To = { (MailAddress)ctor.Invoke(new object[] { null, "[RFax:User", "/FN=0123456789]" }) }
};
There are two tricks here. The first one is to use the internal MailAddress
constructor that doesn't parse/validate the supplied address.
The second trick is to split "fax address" on the @-sign and pass it as two parts (user and domain). This is needed because the SMTP To-header is later written by the framework using the MailAddress.Address
property, and that property returns user + @ + domain.
Some thoughts...
RFC 5322 requires that the email address be in the form local-part@domain. You've omitted the @domain portion.
RFC 5322 further requires that the local-part be a
dot-atom
, consisting of 1 or moreatoms
separated by a single period (e.g.foo
orfoo.bar
). An individualatom
is a sequence of 1 or more of the following characters drawn from the US-ASCII (7-bit) character set of printable US-ASCII characters, excluding 'specials': meaning Upper-/lower-case letters, digits, and the characters! # $ % & ' * + - / = ? ^ _ ` { | } ~
Per the RFC, your email address is absolutely not legal — square brackets are "specials" in the grammar and thus not allowed.
If you want to use something other than a
dot-atom
as yourlocal-part
, then it must be aquoted-string
, defined as a lead-in double-quote ("
), followed byquoted-content
, followed by a lead-out double-quote ("
).quoted-content
is zero or more printable US-ASCII characters in the range 0x21–0x7E, excluding"
and\
.quoted-content
may also include insignificant "folding whitespace".\
,"
and whitespace may be included by escaping them with a\
(e.g., quotes are represented in the quoted string as\"
, backslashes as\\
and spaces as\<sp>
.
Hope this helps!
Edited to Note: Another option would be to send the mail directly via Exchange rather than via its SMTP front end, using the web services exposed by the Exchange Server: http://msdn.microsoft.com/en-us/library/bb204119.aspx.
There are some open source smtp clients out there for .Net. Most are old and stale, but you could just roll your own based on them, such as DotNetOpenMail
So, it appears you're using Diem Mail-to-Fax (PDF manual). Note that the easiest way of doing this, is using the "SMTP IETF Addressing" section - which would simply require an MX record for fax.company.com, and SMTP compliant addresses such as:
fax=0123456789/pn=User@fax.company.com
You would then be able to send these faxes from any client, without having to go through your Exchange server.
The RFAX
scheme requires special support from your Exchange server to route to the fax machine. Since Outlook submits mail via MAPI, it can support this additional address spaces. I'm not entirely sure that even if you can get the SmtpClient
to accept your address, that Exchange will know what to do with it when delivered via SMTP.
I suspect that to use the RFAX scheme, you're going to have to submit the email via MAPI or Web Services, since it's not an SMTP address to begin with.
精彩评论