aboutsummaryrefslogtreecommitdiffstats
path: root/camel/providers/smtp
diff options
context:
space:
mode:
authorJeffrey Stedfast <fejj@ximian.com>2002-01-29 06:10:44 +0800
committerJeffrey Stedfast <fejj@src.gnome.org>2002-01-29 06:10:44 +0800
commit6861836d29bcff5b60ad250da11b0a596a2adccb (patch)
tree71736afd6406a2551927d93cc40a973d6f27f985 /camel/providers/smtp
parent84e6a5344f98f4b64b7316485c3f7a3c8bf8da33 (diff)
downloadgsoc2013-evolution-6861836d29bcff5b60ad250da11b0a596a2adccb.tar
gsoc2013-evolution-6861836d29bcff5b60ad250da11b0a596a2adccb.tar.gz
gsoc2013-evolution-6861836d29bcff5b60ad250da11b0a596a2adccb.tar.bz2
gsoc2013-evolution-6861836d29bcff5b60ad250da11b0a596a2adccb.tar.lz
gsoc2013-evolution-6861836d29bcff5b60ad250da11b0a596a2adccb.tar.xz
gsoc2013-evolution-6861836d29bcff5b60ad250da11b0a596a2adccb.tar.zst
gsoc2013-evolution-6861836d29bcff5b60ad250da11b0a596a2adccb.zip
Use flags rather than a bunch of gboolean variables. (smtp_connect): Same.
2002-01-28 Jeffrey Stedfast <fejj@ximian.com> * providers/smtp/camel-smtp-transport.c (smtp_construct): (connect_to_server): Use flags rather than a bunch of gboolean variables. (smtp_connect): Same. (smtp_mail): Here too. Use the enhanced status codes if available. (smtp_data): And again here. (smtp_helo): Finally here. Also detect the ENHANCEDSTATUSCODES extension. (smtp_rcpt): Use the enhanced status codes if available. (smtp_rset): Here too. (smtp_quit): And finally here. * camel-transport.h: Removed gboolean supports_8bit since this is pretty local to only SMTP for now. svn path=/trunk/; revision=15498
Diffstat (limited to 'camel/providers/smtp')
-rw-r--r--camel/providers/smtp/camel-smtp-transport.c209
-rw-r--r--camel/providers/smtp/camel-smtp-transport.h16
2 files changed, 154 insertions, 71 deletions
diff --git a/camel/providers/smtp/camel-smtp-transport.c b/camel/providers/smtp/camel-smtp-transport.c
index c54e0821d8..0d6851ec11 100644
--- a/camel/providers/smtp/camel-smtp-transport.c
+++ b/camel/providers/smtp/camel-smtp-transport.c
@@ -120,18 +120,18 @@ camel_smtp_transport_class_init (CamelSmtpTransportClass *camel_smtp_transport_c
static void
camel_smtp_transport_init (gpointer object)
{
- CamelTransport *transport = CAMEL_TRANSPORT (object);
+ CamelSmtpTransport *smtp = CAMEL_SMTP_TRANSPORT (object);
- transport->supports_8bit = FALSE;
+ smtp->flags = 0;
}
CamelType
camel_smtp_transport_get_type (void)
{
- static CamelType camel_smtp_transport_type = CAMEL_INVALID_TYPE;
+ static CamelType type = CAMEL_INVALID_TYPE;
- if (camel_smtp_transport_type == CAMEL_INVALID_TYPE) {
- camel_smtp_transport_type =
+ if (type == CAMEL_INVALID_TYPE) {
+ type =
camel_type_register (CAMEL_TRANSPORT_TYPE,
"CamelSmtpTransport",
sizeof (CamelSmtpTransport),
@@ -142,7 +142,7 @@ camel_smtp_transport_get_type (void)
NULL);
}
- return camel_smtp_transport_type;
+ return type;
}
static void
@@ -155,7 +155,7 @@ smtp_construct (CamelService *service, CamelSession *session,
CAMEL_SERVICE_CLASS (parent_class)->construct (service, session, provider, url, ex);
if (camel_url_get_param (url, "use_ssl"))
- smtp_transport->use_ssl = TRUE;
+ smtp_transport->flags |= CAMEL_SMTP_TRANSPORT_USE_SSL;
}
static const char *
@@ -236,7 +236,7 @@ connect_to_server (CamelService *service, CamelException *ex)
struct hostent *h;
guint32 addrlen;
int port, ret;
-
+
if (!CAMEL_SERVICE_CLASS (parent_class)->connect (service, ex))
return FALSE;
@@ -245,14 +245,16 @@ connect_to_server (CamelService *service, CamelException *ex)
return FALSE;
/* set some smtp transport defaults */
- transport->is_esmtp = FALSE;
+ transport->flags &= ~(CAMEL_SMTP_TRANSPORT_IS_ESMTP |
+ CAMEL_SMTP_TRANSPORT_8BITMIME |
+ CAMEL_SMTP_TRANSPORT_ENHANCEDSTATUSCODES);
+
transport->authtypes = NULL;
- CAMEL_TRANSPORT (transport)->supports_8bit = FALSE;
port = service->url->port ? service->url->port : SMTP_PORT;
#if defined(HAVE_NSS) || defined(HAVE_OPENSSL)
- if (transport->use_ssl) {
+ if (transport->flags & CAMEL_SMTP_TRANSPORT_USE_SSL) {
port = service->url->port ? service->url->port : 465;
#ifdef HAVE_NSS
/* use the preferred implementation - NSS */
@@ -268,7 +270,7 @@ connect_to_server (CamelService *service, CamelException *ex)
#endif /* HAVE_NSS || HAVE_OPENSSL */
ret = camel_tcp_stream_connect (CAMEL_TCP_STREAM (tcp_stream), h, port);
- camel_free_host(h);
+ camel_free_host (h);
if (ret == -1) {
camel_exception_setv (ex, CAMEL_EXCEPTION_SERVICE_UNAVAILABLE,
_("Could not connect to %s (port %d): %s"),
@@ -281,7 +283,7 @@ connect_to_server (CamelService *service, CamelException *ex)
/* get the localaddr - needed later by smtp_helo */
addrlen = sizeof (transport->localaddr);
#ifdef HAVE_NSS
- if (transport->use_ssl) {
+ if (transport->flags & CAMEL_SMTP_TRANSPORT_USE_SSL) {
PRFileDesc *sockfd = camel_tcp_stream_get_socket (CAMEL_TCP_STREAM (tcp_stream));
PRNetAddr addr;
char hname[1024];
@@ -320,17 +322,17 @@ connect_to_server (CamelService *service, CamelException *ex)
return FALSE;
}
if (strstr (respbuf, "ESMTP"))
- transport->is_esmtp = TRUE;
+ transport->flags |= CAMEL_SMTP_TRANSPORT_IS_ESMTP;
} while (*(respbuf+3) == '-'); /* if we got "220-" then loop again */
g_free (respbuf);
/* send HELO (or EHLO, depending on the service type) */
- if (!transport->is_esmtp) {
+ if (!(transport->flags & CAMEL_SMTP_TRANSPORT_IS_ESMTP)) {
/* If we did not auto-detect ESMTP, we should still send EHLO */
- transport->is_esmtp = TRUE;
+ transport->flags |= CAMEL_SMTP_TRANSPORT_IS_ESMTP;
if (!smtp_helo (transport, NULL)) {
/* Okay, apprently this server doesn't support ESMTP */
- transport->is_esmtp = FALSE;
+ transport->flags &= ~CAMEL_SMTP_TRANSPORT_IS_ESMTP;
smtp_helo (transport, ex);
}
} else {
@@ -375,7 +377,8 @@ smtp_connect (CamelService *service, CamelException *ex)
gboolean authenticated = FALSE;
char *errbuf = NULL;
- if (!transport->is_esmtp || !g_hash_table_lookup (transport->authtypes, service->url->authmech)) {
+ if (!(transport->flags & CAMEL_SMTP_TRANSPORT_IS_ESMTP) ||
+ !g_hash_table_lookup (transport->authtypes, service->url->authmech)) {
camel_exception_setv (ex, CAMEL_EXCEPTION_SERVICE_CANT_AUTHENTICATE,
_("SMTP server %s does not support requested "
"authentication type %s"), service->url->host,
@@ -545,6 +548,7 @@ query_auth_types (CamelService *service, CamelException *ex)
}
smtp_disconnect (service, TRUE, NULL);
+
return types;
}
@@ -601,8 +605,7 @@ smtp_send_to (CamelTransport *transport, CamelMedium *message,
if (!recipients) {
camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
- _("Cannot send message: "
- "no recipients defined."));
+ _("Cannot send message: no recipients defined."));
camel_operation_end (NULL);
return FALSE;
}
@@ -664,20 +667,36 @@ smtp_send (CamelTransport *transport, CamelMedium *message, CamelException *ex)
return status;
}
+static const char *
+smtp_next_token (const char *buf)
+{
+ const unsigned char *token;
+
+ token = (const unsigned char *) buf;
+ while (*token && !isspace ((int) *token))
+ token++;
+
+ while (*token && isspace ((int) *token))
+ token++;
+
+ return (const char *) token;
+}
+
static gboolean
smtp_helo (CamelSmtpTransport *transport, CamelException *ex)
{
/* say hello to the server */
- gchar *cmdbuf, *respbuf = NULL;
+ char *cmdbuf, *respbuf = NULL;
+ const char *token;
struct hostent *host;
camel_operation_start_transient (NULL, _("SMTP Greeting"));
/* get the local host name */
- host = gethostbyaddr ((gchar *)&transport->localaddr.sin_addr, sizeof (transport->localaddr.sin_addr), AF_INET);
+ host = gethostbyaddr ((char *)&transport->localaddr.sin_addr, sizeof (transport->localaddr.sin_addr), AF_INET);
/* hiya server! how are you today? */
- if (transport->is_esmtp) {
+ if (transport->flags & CAMEL_SMTP_TRANSPORT_IS_ESMTP) {
if (host && host->h_name)
cmdbuf = g_strdup_printf ("EHLO %s\r\n", host->h_name);
else
@@ -719,17 +738,28 @@ smtp_helo (CamelSmtpTransport *transport, CamelException *ex)
return FALSE;
}
- if (strstrcase (respbuf, "8BITMIME")) {
- d(fprintf (stderr, "This server supports 8bit MIME\n"));
- CAMEL_TRANSPORT (transport)->supports_8bit = TRUE;
- }
+ token = respbuf + 4;
- /* Only parse authtypes if we don't already have them */
- if (transport->is_esmtp && strstr (respbuf, "AUTH") && !transport->authtypes) {
- /* parse for supported AUTH types */
- char *auths = strstr (respbuf, "AUTH") + 4;
-
- transport->authtypes = esmtp_get_authtypes (auths);
+ if (transport->flags & CAMEL_SMTP_TRANSPORT_IS_ESMTP) {
+ if (!strncmp (token, "8BITMIME", 8)) {
+ d(fprintf (stderr, "This server supports 8bit MIME\n"));
+ transport->flags |= CAMEL_SMTP_TRANSPORT_8BITMIME;
+ } else if (!strncmp (token, "ENHANCEDSTATUSCODES", 19)) {
+ d(fprintf (stderr, "This server supports enhanced status codes\n"));
+ transport->flags |= CAMEL_SMTP_TRANSPORT_ENHANCEDSTATUSCODES;
+ } else if (!transport->authtypes && !strncmp (token, "AUTH", 4)) {
+ /* Don't bother parsing any authtypes if we already have a list.
+ * Some servers will list AUTH twice, once the standard way and
+ * once the way Microsoft Outlook requires them to be:
+ *
+ * 250-AUTH LOGIN PLAIN DIGEST-MD5 CRAM-MD5
+ * 250-AUTH=LOGIN PLAIN DIGEST-MD5 CRAM-MD5
+ **/
+
+ /* parse for supported AUTH types */
+ token = smtp_next_token (token);
+ transport->authtypes = esmtp_get_authtypes (token);
+ }
}
} while (*(respbuf+3) == '-'); /* if we got "250-" then loop again */
g_free (respbuf);
@@ -742,7 +772,7 @@ smtp_helo (CamelSmtpTransport *transport, CamelException *ex)
static gboolean
smtp_auth (CamelSmtpTransport *transport, const char *mech, CamelException *ex)
{
- gchar *cmdbuf, *respbuf = NULL, *challenge;
+ char *cmdbuf, *respbuf = NULL, *challenge;
CamelSasl *sasl = NULL;
camel_operation_start_transient (NULL, _("SMTP Authentication"));
@@ -848,10 +878,10 @@ static gboolean
smtp_mail (CamelSmtpTransport *transport, const char *sender, gboolean has_8bit_parts, CamelException *ex)
{
/* we gotta tell the smtp server who we are. (our email addy) */
- gchar *cmdbuf, *respbuf = NULL;
+ char *cmdbuf, *respbuf = NULL;
/* enclose address in <>'s since some SMTP daemons *require* that */
- if (CAMEL_TRANSPORT (transport)->supports_8bit && has_8bit_parts)
+ if (transport->flags & CAMEL_SMTP_TRANSPORT_8BITMIME && has_8bit_parts)
cmdbuf = g_strdup_printf ("MAIL FROM: <%s> BODY=8BITMIME\r\n", sender);
else
cmdbuf = g_strdup_printf ("MAIL FROM: <%s>\r\n", sender);
@@ -875,13 +905,21 @@ smtp_mail (CamelSmtpTransport *transport, const char *sender, gboolean has_8bit_
d(fprintf (stderr, "received: %s\n", respbuf ? respbuf : "(null)"));
if (!respbuf || strncmp (respbuf, "250", 3)) {
+ const char *token;
int error;
- error = respbuf ? atoi (respbuf) : 0;
- g_free (respbuf);
+ if (!respbuf || !(transport->flags & CAMEL_SMTP_TRANSPORT_ENHANCEDSTATUSCODES)) {
+ error = respbuf ? atoi (respbuf) : 0;
+ token = get_smtp_error_string (error);
+ } else
+ token = respbuf + 4;
+
camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
- _("MAIL FROM response error: %s: mail not sent"),
- get_smtp_error_string (error));
+ _("MAIL FROM response error: %s"),
+ token);
+
+ g_free (respbuf);
+
return FALSE;
}
} while (*(respbuf+3) == '-'); /* if we got "250-" then loop again */
@@ -895,7 +933,7 @@ smtp_rcpt (CamelSmtpTransport *transport, const char *recipient, CamelException
{
/* we gotta tell the smtp server who we are going to be sending
* our email to */
- gchar *cmdbuf, *respbuf = NULL;
+ char *cmdbuf, *respbuf = NULL;
/* enclose address in <>'s since some SMTP daemons *require* that */
cmdbuf = g_strdup_printf ("RCPT TO: <%s>\r\n", recipient);
@@ -912,20 +950,28 @@ smtp_rcpt (CamelSmtpTransport *transport, const char *recipient, CamelException
g_free (cmdbuf);
do {
- /* Check for "250 Sender OK..." */
+ /* Check for "250 Recipient OK..." */
g_free (respbuf);
respbuf = camel_stream_buffer_read_line (CAMEL_STREAM_BUFFER (transport->istream));
d(fprintf (stderr, "received: %s\n", respbuf ? respbuf : "(null)"));
if (!respbuf || strncmp (respbuf, "250", 3)) {
+ const char *token;
int error;
- error = respbuf ? atoi (respbuf) : 0;
- g_free (respbuf);
+ if (!respbuf || !(transport->flags & CAMEL_SMTP_TRANSPORT_ENHANCEDSTATUSCODES)) {
+ error = respbuf ? atoi (respbuf) : 0;
+ token = get_smtp_error_string (error);
+ } else
+ token = respbuf + 4;
+
camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
- _("RCPT TO response error: %s: mail not sent"),
- get_smtp_error_string (error));
+ _("RCPT TO response error: %s"),
+ token);
+
+ g_free (respbuf);
+
return FALSE;
}
} while (*(respbuf+3) == '-'); /* if we got "250-" then loop again */
@@ -948,7 +994,7 @@ smtp_data (CamelSmtpTransport *transport, CamelMedium *message, gboolean has_8bi
/* if the message contains 8bit mime parts and the server
doesn't support it, encode 8bit parts to the best
encoding. This will also enforce an encoding to keep the lines in limit */
- if (has_8bit_parts && !CAMEL_TRANSPORT (transport)->supports_8bit)
+ if (has_8bit_parts && !(transport->flags & CAMEL_SMTP_TRANSPORT_8BITMIME))
camel_mime_message_encode_8bit_parts (CAMEL_MIME_MESSAGE (message));
cmdbuf = g_strdup ("DATA\r\n");
@@ -972,13 +1018,21 @@ smtp_data (CamelSmtpTransport *transport, CamelMedium *message, gboolean has_8bi
/* we should have gotten instructions on how to use the DATA command:
* 354 Enter mail, end with "." on a line by itself
*/
+ const char *token;
int error;
-
- error = respbuf ? atoi (respbuf) : 0;
- g_free (respbuf);
+
+ if (!respbuf || !(transport->flags & CAMEL_SMTP_TRANSPORT_ENHANCEDSTATUSCODES)) {
+ error = respbuf ? atoi (respbuf) : 0;
+ token = get_smtp_error_string (error);
+ } else
+ token = respbuf + 4;
+
camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
- _("DATA response error: %s: mail not sent"),
- get_smtp_error_string (error));
+ _("DATA response error: %s"),
+ token);
+
+ g_free (respbuf);
+
return FALSE;
}
@@ -1049,14 +1103,21 @@ smtp_data (CamelSmtpTransport *transport, CamelMedium *message, gboolean has_8bi
d(fprintf (stderr, "received: %s\n", respbuf ? respbuf : "(null)"));
if (!respbuf || strncmp (respbuf, "250", 3)) {
+ const char *token;
int error;
- error = respbuf ? atoi (respbuf) : 0;
- g_free (respbuf);
+ if (!respbuf || !(transport->flags & CAMEL_SMTP_TRANSPORT_ENHANCEDSTATUSCODES)) {
+ error = respbuf ? atoi (respbuf) : 0;
+ token = get_smtp_error_string (error);
+ } else
+ token = respbuf + 4;
+
camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
- _("DATA response error: message termination: "
- "%s: mail not sent"),
- get_smtp_error_string (error));
+ _("DATA termination response error: %s"),
+ token);
+
+ g_free (respbuf);
+
return FALSE;
}
} while (*(respbuf+3) == '-'); /* if we got "250-" then loop again */
@@ -1069,7 +1130,7 @@ static gboolean
smtp_rset (CamelSmtpTransport *transport, CamelException *ex)
{
/* we are going to reset the smtp server (just to be nice) */
- gchar *cmdbuf, *respbuf = NULL;
+ char *cmdbuf, *respbuf = NULL;
cmdbuf = g_strdup ("RSET\r\n");
@@ -1092,13 +1153,21 @@ smtp_rset (CamelSmtpTransport *transport, CamelException *ex)
d(fprintf (stderr, "received: %s\n", respbuf ? respbuf : "(null)"));
if (!respbuf || strncmp (respbuf, "250", 3)) {
+ const char *token;
int error;
- error = respbuf ? atoi (respbuf) : 0;
- g_free (respbuf);
+ if (!respbuf || !(transport->flags & CAMEL_SMTP_TRANSPORT_ENHANCEDSTATUSCODES)) {
+ error = respbuf ? atoi (respbuf) : 0;
+ token = get_smtp_error_string (error);
+ } else
+ token = respbuf + 4;
+
camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
_("RSET response error: %s"),
- get_smtp_error_string (error));
+ token);
+
+ g_free (respbuf);
+
return FALSE;
}
} while (*(respbuf+3) == '-'); /* if we got "250-" then loop again */
@@ -1111,7 +1180,7 @@ static gboolean
smtp_quit (CamelSmtpTransport *transport, CamelException *ex)
{
/* we are going to reset the smtp server (just to be nice) */
- gchar *cmdbuf, *respbuf = NULL;
+ char *cmdbuf, *respbuf = NULL;
cmdbuf = g_strdup ("QUIT\r\n");
@@ -1134,13 +1203,21 @@ smtp_quit (CamelSmtpTransport *transport, CamelException *ex)
d(fprintf (stderr, "received: %s\n", respbuf ? respbuf : "(null)"));
if (!respbuf || strncmp (respbuf, "221", 3)) {
+ const char *token;
int error;
- error = respbuf ? atoi (respbuf) : 0;
- g_free (respbuf);
+ if (!respbuf || !(transport->flags & CAMEL_SMTP_TRANSPORT_ENHANCEDSTATUSCODES)) {
+ error = respbuf ? atoi (respbuf) : 0;
+ token = get_smtp_error_string (error);
+ } else
+ token = respbuf + 4;
+
camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
- _("QUIT response error: %s: non-fatal"),
- get_smtp_error_string (error));
+ _("QUIT response error: %s"),
+ token);
+
+ g_free (respbuf);
+
return FALSE;
}
} while (*(respbuf+3) == '-'); /* if we got "221-" then loop again */
diff --git a/camel/providers/smtp/camel-smtp-transport.h b/camel/providers/smtp/camel-smtp-transport.h
index 7a6b631c46..8bba00e987 100644
--- a/camel/providers/smtp/camel-smtp-transport.h
+++ b/camel/providers/smtp/camel-smtp-transport.h
@@ -47,15 +47,21 @@ extern "C" {
#define CAMEL_IS_SMTP_TRANSPORT(o) (CAMEL_CHECK_TYPE((o), CAMEL_SMTP_TRANSPORT_TYPE))
+#define CAMEL_SMTP_TRANSPORT_IS_ESMTP (1 << 0)
+#define CAMEL_SMTP_TRANSPORT_8BITMIME (1 << 1)
+#define CAMEL_SMTP_TRANSPORT_ENHANCEDSTATUSCODES (1 << 2)
+
+#define CAMEL_SMTP_TRANSPORT_USE_SSL (1 << 3)
+
typedef struct {
CamelTransport parent_object;
-
+
CamelStream *istream, *ostream;
-
- gboolean use_ssl, is_esmtp;
-
+
+ guint32 flags;
+
struct sockaddr_in localaddr;
-
+
GHashTable *authtypes;
} CamelSmtpTransport;