| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549 |
- //
- // System.Net.Mail.SmtpClient.cs
- //
- // Author:
- // Tim Coleman ([email protected])
- //
- // Copyright (C) Tim Coleman, 2004
- //
- //
- // Permission is hereby granted, free of charge, to any person obtaining
- // a copy of this software and associated documentation files (the
- // "Software"), to deal in the Software without restriction, including
- // without limitation the rights to use, copy, modify, merge, publish,
- // distribute, sublicense, and/or sell copies of the Software, and to
- // permit persons to whom the Software is furnished to do so, subject to
- // the following conditions:
- //
- // The above copyright notice and this permission notice shall be
- // included in all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
- // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
- // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
- // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
- // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
- // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
- //
- #if NET_2_0
- using System;
- using System.Collections.Generic;
- using System.ComponentModel;
- using System.IO;
- using System.Net;
- using System.Net.Mime;
- using System.Net.Sockets;
- using System.Security.Cryptography.X509Certificates;
- using System.Text;
- using System.Threading;
- namespace System.Net.Mail {
- public class SmtpClient
- {
- #region Fields
- string host;
- int port;
- int timeout = 100000;
- ICredentialsByHost credentials;
- bool useDefaultCredentials;
- string pickupDirectoryLocation;
- SmtpDeliveryMethod deliveryMethod;
- bool enableSsl;
- X509CertificateCollection clientCertificates;
- TcpClient client;
- NetworkStream stream;
- StreamWriter writer;
- StreamReader reader;
- int boundaryIndex;
- Mutex mutex = new Mutex ();
- const string MimeVersion = "1.0 (produced by Mono System.Net.Mail.SmtpClient)";
- #endregion // Fields
- #region Constructors
- public SmtpClient ()
- : this (null, 0)
- {
- }
- public SmtpClient (string host)
- : this (host, 0)
- {
- }
- [MonoTODO ("Load default settings from configuration.")]
- public SmtpClient (string host, int port)
- {
- // FIXME: load from configuration
- if (String.IsNullOrEmpty (host))
- Host = "127.0.0.1";
- else
- Host = host;
-
- // FIXME: load from configuration
- if (port == 0)
- Port = 25;
- else
- Port = port;
- // FIXME: load credentials from configuration
- }
- #endregion // Constructors
- #region Properties
- [MonoTODO]
- public X509CertificateCollection ClientCertificates {
- get { return clientCertificates; }
- }
- public ICredentialsByHost Credentials {
- get { return credentials; }
- set { credentials = value; }
- }
- public SmtpDeliveryMethod DeliveryMethod {
- get { return deliveryMethod; }
- set { deliveryMethod = value; }
- }
- public bool EnableSsl {
- get { return enableSsl; }
- set { enableSsl = value; }
- }
- public string Host {
- get { return host; }
- [MonoTODO ("Check to make sure an email is not being sent.")]
- set {
- if (value == null)
- throw new ArgumentNullException ();
- if (value.Length == 0)
- throw new ArgumentException ();
- host = value;
- }
- }
- public string PickupDirectoryLocation {
- get { return pickupDirectoryLocation; }
- set { pickupDirectoryLocation = value; }
- }
- public int Port {
- get { return port; }
- [MonoTODO ("Check to make sure an email is not being sent.")]
- set {
- if (value <= 0)
- throw new ArgumentOutOfRangeException ();
- port = value;
- }
- }
- [MonoTODO]
- public ServicePoint ServicePoint {
- get { throw new NotImplementedException (); }
- }
- public int Timeout {
- get { return timeout; }
- [MonoTODO ("Check to make sure an email is not being sent.")]
- set {
- if (value < 0)
- throw new ArgumentOutOfRangeException ();
- timeout = value;
- }
- }
- [MonoTODO]
- public bool UseDefaultCredentials {
- get { return useDefaultCredentials; }
- set { useDefaultCredentials = value; }
- }
- #endregion // Properties
- #region Events
- public event SendCompletedEventHandler SendCompleted;
- #endregion // Events
- #region Methods
- private void EndSection (string section)
- {
- SendData (String.Format ("--{0}--", section));
- }
- private string GenerateBoundary ()
- {
- string output = GenerateBoundary (boundaryIndex);
- boundaryIndex += 1;
- return output;
- }
- private static string GenerateBoundary (int index)
- {
- return String.Format ("--boundary_{0}_{1}", index, Guid.NewGuid ().ToString ("D"));
- }
- private bool IsError (SmtpResponse status)
- {
- return ((int) status.StatusCode) >= 400;
- }
- protected void OnSendCompleted (AsyncCompletedEventArgs e)
- {
- if (SendCompleted != null)
- SendCompleted (this, e);
- }
- private SmtpResponse Read ()
- {
- SmtpResponse response;
- char[] buf = new char [3];
- reader.Read (buf, 0, 3);
- reader.Read ();
- response.StatusCode = (SmtpStatusCode) Int32.Parse (new String (buf));
- response.Description = reader.ReadLine ();
- return response;
- }
- [MonoTODO ("Need to work on message attachments.")]
- public void Send (MailMessage message)
- {
- // Block while sending
- mutex.WaitOne ();
- SmtpResponse status;
- client = new TcpClient (host, port);
- stream = client.GetStream ();
- writer = new StreamWriter (stream);
- reader = new StreamReader (stream);
- boundaryIndex = 0;
- string boundary = GenerateBoundary ();
- bool hasAlternateViews = (message.AlternateViews.Count > 0);
- bool hasAttachments = (message.Attachments.Count > 0);
- status = Read ();
- if (IsError (status))
- throw new SmtpException (status.StatusCode);
- // HELO
- status = SendCommand (Command.Helo, Dns.GetHostName ());
- if (IsError (status))
- throw new SmtpException (status.StatusCode);
- // MAIL FROM:
- status = SendCommand (Command.MailFrom, message.From.Address);
- if (IsError (status))
- throw new SmtpException (status.StatusCode);
- // Send RCPT TO: for all recipients
- List<SmtpFailedRecipientException> sfre = new List<SmtpFailedRecipientException> ();
- for (int i = 0; i < message.To.Count; i ++) {
- status = SendCommand (Command.RcptTo, message.To [i].Address);
- if (IsError (status))
- sfre.Add (new SmtpFailedRecipientException (status.StatusCode, message.To [i].Address.ToString ()));
- }
- for (int i = 0; i < message.CC.Count; i ++) {
- status = SendCommand (Command.RcptTo, message.CC [i].Address);
- if (IsError (status))
- sfre.Add (new SmtpFailedRecipientException (status.StatusCode, message.CC [i].Address.ToString ()));
- }
- for (int i = 0; i < message.Bcc.Count; i ++) {
- status = SendCommand (Command.RcptTo, message.Bcc [i].Address);
- if (IsError (status))
- sfre.Add (new SmtpFailedRecipientException (status.StatusCode, message.Bcc [i].Address.ToString ()));
- }
- if (sfre.Count > 0)
- throw new SmtpFailedRecipientsException ("failed recipients", sfre.ToArray ());
- // DATA
- status = SendCommand (Command.Data);
- if (IsError (status))
- throw new SmtpException (status.StatusCode);
- // Figure out the message content type
- ContentType messageContentType = message.BodyContentType;
- if (hasAttachments || hasAlternateViews) {
- messageContentType.Boundary = boundary;
- if (hasAttachments)
- messageContentType.MediaType = "multipart/mixed";
- else
- messageContentType.MediaType = "multipart/alternative";
- }
- // Send message headers
- SendHeader (HeaderName.From, message.From.ToString ());
- SendHeader (HeaderName.To, message.To.ToString ());
- if (message.CC.Count > 0)
- SendHeader (HeaderName.Cc, message.CC.ToString ());
- if (message.Bcc.Count > 0)
- SendHeader (HeaderName.Bcc, message.Bcc.ToString ());
- SendHeader (HeaderName.Subject, message.Subject);
- foreach (string s in message.Headers.AllKeys)
- SendHeader (s, message.Headers [s]);
- SendHeader ("Content-Type", messageContentType.ToString ());
- SendData ("");
- if (hasAlternateViews) {
- string innerBoundary = boundary;
- // The body is *technically* an alternative view. The body text goes FIRST because
- // that is most compatible with non-MIME readers.
- //
- // If there are attachments, then the main content-type is multipart/mixed and
- // the subpart has type multipart/alternative. Then all of the views have their
- // own types.
- //
- // If there are no attachments, then the main content-type is multipart/alternative
- // and we don't need this subpart.
- if (hasAttachments) {
- innerBoundary = GenerateBoundary ();
- ContentType contentType = new ContentType ("multipart/alternative");
- contentType.Boundary = innerBoundary;
- StartSection (boundary, contentType);
- }
-
- // Start the section for the body text. This is either section "1" or "0" depending
- // on whether there are attachments.
- StartSection (innerBoundary, messageContentType, TransferEncoding.QuotedPrintable);
- SendData (message.Body);
- // Send message attachments.
- SendAttachments (message.Attachments, innerBoundary);
- if (hasAttachments)
- EndSection (innerBoundary);
- }
- else {
- // If this is multipart then we need to send a boundary before the body.
- if (hasAttachments) {
- // FIXME: check this
- ContentType contentType = new ContentType ("multipart/alternative");
- StartSection (boundary, contentType, TransferEncoding.QuotedPrintable);
- }
- SendData (message.Body);
- }
- // Send attachments
- if (hasAttachments) {
- string innerBoundary = boundary;
- // If we have alternate views and attachments then we need to nest this part inside another
- // boundary. Otherwise, we are cool with the boundary we have.
- if (hasAlternateViews) {
- innerBoundary = GenerateBoundary ();
- ContentType contentType = new ContentType ("multipart/mixed");
- contentType.Boundary = innerBoundary;
- StartSection (boundary, contentType);
- }
- SendAttachments (message.Attachments, innerBoundary);
- if (hasAlternateViews)
- EndSection (innerBoundary);
- }
- SendData (".");
- status = Read ();
- if (IsError (status))
- throw new SmtpException (status.StatusCode);
- status = SendCommand (Command.Quit);
- writer.Close ();
- reader.Close ();
- stream.Close ();
- client.Close ();
- // Release the mutex to allow other threads access
- mutex.ReleaseMutex ();
- }
- public void Send (string from, string to, string subject, string body)
- {
- Send (new MailMessage (from, to, subject, body));
- }
- private void SendData (string data)
- {
- writer.WriteLine (data);
- writer.Flush ();
- }
- [MonoTODO]
- public void SendAsync (MailMessage message, object userToken)
- {
- Send (message);
- OnSendCompleted (new AsyncCompletedEventArgs (null, false, userToken));
- }
- public void SendAsync (string from, string to, string subject, string body, object userToken)
- {
- SendAsync (new MailMessage (from, to, subject, body), userToken);
- }
- [MonoTODO]
- public void SendAsyncCancel ()
- {
- throw new NotImplementedException ();
- }
- private void SendAttachments (AttachmentCollection attachments, string boundary)
- {
- for (int i = 0; i < attachments.Count; i += 1) {
- // FIXME: check this
- ContentType contentType = new ContentType ("multipart/alternative");
- StartSection (boundary, contentType, attachments [i].TransferEncoding);
- switch (attachments [i].TransferEncoding) {
- case TransferEncoding.Base64:
- byte[] content = new byte [attachments [i].ContentStream.Length];
- attachments [i].ContentStream.Read (content, 0, content.Length);
- SendData (Convert.ToBase64String (content, Base64FormattingOptions.InsertLineBreaks));
- break;
- case TransferEncoding.QuotedPrintable:
- StreamReader sr = new StreamReader (attachments [i].ContentStream);
- SendData (ToQuotedPrintable (sr.ReadToEnd ()));
- break;
- //case TransferEncoding.SevenBit:
- //case TransferEncoding.Unknown:
- default:
- SendData ("TO BE IMPLEMENTED");
- break;
- }
- }
- }
- private SmtpResponse SendCommand (string command, string data)
- {
- writer.Write (command);
- writer.Write (" ");
- SendData (data);
- return Read ();
- }
- private SmtpResponse SendCommand (string command)
- {
- writer.WriteLine (command);
- writer.Flush ();
- return Read ();
- }
- private void SendHeader (string name, string value)
- {
- SendData (String.Format ("{0}: {1}", name, value));
- }
- private void StartSection (string section, ContentType sectionContentType)
- {
- SendData (String.Format ("--{0}", section));
- SendHeader ("content-type", sectionContentType.ToString ());
- SendData ("");
- }
- private void StartSection (string section, ContentType sectionContentType,TransferEncoding transferEncoding)
- {
- SendData (String.Format ("--{0}", section));
- SendHeader ("content-type", sectionContentType.ToString ());
- SendHeader ("content-transfer-encoding", GetTransferEncodingName (transferEncoding));
- SendData ("");
- }
- private string ToQuotedPrintable (string input)
- {
- StringReader reader = new StringReader (input);
- StringWriter writer = new StringWriter ();
- int i;
- while ((i = reader.Read ()) > 0) {
- if (i > 127) {
- writer.Write ("=");
- writer.Write (Convert.ToString (i, 16).ToUpper ());
- }
- else
- writer.Write (Convert.ToChar (i));
- }
- return writer.GetStringBuilder ().ToString ();
- }
- private static string GetTransferEncodingName (TransferEncoding encoding)
- {
- switch (encoding) {
- case TransferEncoding.QuotedPrintable:
- return "quoted-printable";
- case TransferEncoding.SevenBit:
- return "7bit";
- case TransferEncoding.Base64:
- return "base64";
- }
- return "unknown";
- }
- /*
- [MonoTODO]
- private sealed ContextAwareResult IGetContextAwareResult.GetContextAwareResult ()
- {
- throw new NotImplementedException ();
- }
- */
- #endregion // Methods
- // The Command struct is used to store constant string values representing SMTP commands.
- private struct Command {
- public const string Data = "DATA";
- public const string Helo = "HELO";
- public const string MailFrom = "MAIL FROM:";
- public const string Quit = "QUIT";
- public const string RcptTo = "RCPT TO:";
- }
- // The HeaderName struct is used to store constant string values representing mail headers.
- private struct HeaderName {
- public const string ContentTransferEncoding = "Content-Transfer-Encoding";
- public const string ContentType = "Content-Type";
- public const string Bcc = "Bcc";
- public const string Cc = "Cc";
- public const string From = "From";
- public const string Subject = "Subject";
- public const string To = "To";
- public const string MimeVersion = "MIME-Version";
- public const string MessageId = "Message-ID";
- }
- // This object encapsulates the status code and description of an SMTP response.
- private struct SmtpResponse {
- public SmtpStatusCode StatusCode;
- public string Description;
- }
- }
- }
- #endif // NET_2_0
|