diff options
-rw-r--r-- | camel/ChangeLog | 30 | ||||
-rw-r--r-- | camel/providers/imap4/camel-imap-engine.c | 25 | ||||
-rw-r--r-- | camel/providers/imap4/camel-imap-specials.c | 6 | ||||
-rw-r--r-- | camel/providers/imap4/camel-imap-store.c | 827 | ||||
-rw-r--r-- | camel/providers/imap4/camel-imap-store.h | 61 |
5 files changed, 914 insertions, 35 deletions
diff --git a/camel/ChangeLog b/camel/ChangeLog index 919c647fc9..b73acb5484 100644 --- a/camel/ChangeLog +++ b/camel/ChangeLog @@ -1,5 +1,18 @@ 2004-03-25 Jeffrey Stedfast <fejj@ximian.com> + * providers/imap4/camel-imap-engine.c + (camel_imap_engine_parse_resp_code): No longer need to split ']' + tokens from atom tokens due to a fixup in the ABNF grammar in + rfc3501. + + * providers/imap4/camel-imap-specials.c: Changed ATOM_SPECIALS to + include ']' (this is an addition in rfc3501). + + * providers/imap4/camel-imap-store.[c,h]: New Store class for + IMAP. Implemnted a bunch of but still got a ways to go. + +2004-03-25 Jeffrey Stedfast <fejj@ximian.com> + * providers/imap/camel-imap-folder.c (imap_update_summary): If the server is imap.web.de, just ask for all the headers rather than "HEADER.FIELDS.NOT (RECEIVED)". Actually, maybe we should always @@ -31,14 +44,15 @@ from the provider description 2004-03-25 Sivaiah Nallagatla <snallagatla@novell.com> - - * providers/groupwise/camel-groupwise-provider.c : removed - ldap seetings and added a settng for SOAP port - * providers/groupwise/camel-gw-listner.h (add_calendar_tasks_sources) - (remove_calendar_tasks_sources), (modify_calendar_tasks_sources) : read port number - from url instead of hardcoding - Also removed code for adding e-sources for ldap address book and - adding now e-sources for groupwise address book + + * providers/groupwise/camel-groupwise-provider.c: removed ldap + seetings and added a settng for SOAP port + + * providers/groupwise/camel-gw-listner.h + (add_calendar_tasks_sources, remove_calendar_tasks_sources), + (modify_calendar_tasks_sources): read port number from url instead + of hardcoding. Also removed code for adding e-sources for ldap + address book and adding now e-sources for groupwise address book 2004-03-24 Jeffrey Stedfast <fejj@ximian.com> diff --git a/camel/providers/imap4/camel-imap-engine.c b/camel/providers/imap4/camel-imap-engine.c index e7d6ce5899..d82bdd43d9 100644 --- a/camel/providers/imap4/camel-imap-engine.c +++ b/camel/providers/imap4/camel-imap-engine.c @@ -887,14 +887,6 @@ camel_imap_engine_parse_resp_code (CamelIMAPEngine *engine, CamelException *ex) return -1; } - if (token.v.atom[strlen (token.v.atom) - 1] == ']') { - /* split this atom ("FOO]") into 2 tokens: "FOO" and "]" */ - token.token = ']'; - token.v.atom[strlen (token.v.atom) - 1] = '\0'; - camel_imap_stream_unget_token (engine->istream, &token); - token.token = CAMEL_IMAP_TOKEN_ATOM; - } - for (code = 0; imap_resp_codes[code].name; code++) { if (!strcmp (imap_resp_codes[code].name, token.v.atom)) { if (engine->current && imap_resp_codes[code].save) { @@ -1054,14 +1046,6 @@ camel_imap_engine_parse_resp_code (CamelIMAPEngine *engine, CamelException *ex) goto exception; } - if (token.v.atom[strlen (token.v.atom) - 1] == ']') { - /* this should be the case, but if not - no big. */ - token.token = ']'; - token.v.atom[strlen (token.v.atom) - 1] = '\0'; - camel_imap_stream_unget_token (engine->istream, &token); - token.token = CAMEL_IMAP_TOKEN_ATOM; - } - if (resp != NULL) resp->v.copyuid.destset = g_strdup (token.v.atom); @@ -1071,17 +1055,10 @@ camel_imap_engine_parse_resp_code (CamelIMAPEngine *engine, CamelException *ex) /* extensions are of the form: "[" atom [SPACE 1*<any TEXT_CHAR except "]">] "]" */ - /* eat up the TEXT_CHARs, being careful to check atoms for a trailing ']' */ + /* eat up the TEXT_CHARs */ while (token.token != ']' && token.token != '\n') { if (camel_imap_engine_next_token (engine, &token, ex) == -1) goto exception; - - if (token.token == CAMEL_IMAP_TOKEN_ATOM) { - if (token.v.atom[strlen (token.v.atom) - 1] == ']') { - token.token = ']'; - break; - } - } } break; diff --git a/camel/providers/imap4/camel-imap-specials.c b/camel/providers/imap4/camel-imap-specials.c index 5e61d1776b..5289394f19 100644 --- a/camel/providers/imap4/camel-imap-specials.c +++ b/camel/providers/imap4/camel-imap-specials.c @@ -26,18 +26,18 @@ #include "camel-imap-specials.h" -#define CHARS_ATOM_SPECIALS "(){" +#define CHARS_ATOM_SPECIALS "(){]" #define CHARS_LWSP " \t\r\n" #define CHARS_QUOTED_SPECIALS "\\\"" #define CHARS_LIST_WILDCARDS "*%" unsigned char camel_imap_specials[256] = { - 2, 2, 2, 2, 2, 2, 2, 2, 2, 6, 6, 2, 2, 6, 2, 2, + 2, 2, 2, 2, 2, 2, 2, 2, 2, 6, 6, 2, 2, 6, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 20, 0, 8, 0, 0, 32, 0, 0, 1, 1, 32, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 8, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, diff --git a/camel/providers/imap4/camel-imap-store.c b/camel/providers/imap4/camel-imap-store.c new file mode 100644 index 0000000000..10019164ad --- /dev/null +++ b/camel/providers/imap4/camel-imap-store.c @@ -0,0 +1,827 @@ +/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ +/* Camel + * Copyright (C) 1999-2004 Jeffrey Stedfast + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Street #330, Boston, MA 02111-1307, USA. + */ + + +#ifdef HAVE_CONFIG_H +#include <config.h> +#endif + +#include <stdio.h> +#include <string.h> +#include <ctype.h> + +#include <camel/camel-sasl.h> +#include <camel/camel-tcp-stream-raw.h> +#include <camel/camel-tcp-stream-ssl.h> + +#include "camel-imap-store.h" +#include "camel-imap-engine.h" +#include "camel-imap-folder.h" +#include "camel-imap-stream.h" +#include "camel-imap-command.h" + + +static void camel_imap_store_class_init (CamelIMAPStoreClass *klass); +static void camel_imap_store_init (CamelIMAPStore *store, CamelIMAPStoreClass *klass); +static void camel_imap_store_finalize (CamelObject *object); + +/* service methods */ +static void imap_construct (CamelService *service, CamelSession *session, + CamelProvider *provider, CamelURL *url, + CamelException *ex); +static char *imap_get_name (CamelService *service, gboolean brief); +static gboolean imap_connect (CamelService *service, CamelException *ex); +static gboolean imap_disconnect (CamelService *service, gboolean clean, CamelException *ex); +static GList *imap_query_auth_types (CamelService *service, CamelException *ex); + +static guint imap_hash_folder_name (gconstpointer key); +static gint imap_compare_folder_name (gconstpointer a, gconstpointer b); + +/* store methods */ +static CamelFolder *imap_get_folder (CamelStore *store, const char *folder_name, guint32 flags, CamelException *ex); +static CamelFolderInfo *imap_create_folder (CamelStore *store, const char *parent_name, const char *folder_name, CamelException *ex); +static void imap_delete_folder (CamelStore *store, const char *folder_name, CamelException *ex); +static void imap_rename_folder (CamelStore *store, const char *old_name, const char *new_name, CamelException *ex); +static void imap_sync (CamelStore *store, gboolean expunge, CamelException *ex); +static CamelFolderInfo *imap_get_folder_info (CamelStore *store, const char *top, guint32 flags, CamelException *ex); +static void imap_subscribe_folder (CamelStore *store, const char *folder_name, CamelException *ex); +static void imap_unsubscribe_folder (CamelStore *store, const char *folder_name, CamelException *ex); +static void imap_noop (CamelStore *store, CamelException *ex); + + +static CamelStoreClass *parent_class = NULL; + + +CamelType +camel_imap_store_get_type (void) +{ + static CamelType type = 0; + + if (!type) { + type = camel_type_register (CAMEL_TYPE_IMAP_STORE, + "CamelIMAPStore", + sizeof (CamelIMAPStore), + sizeof (CamelIMAPStoreClass), + (CamelObjectClassInitFunc) camel_imap_store_class_init, + NULL, + (CamelObjectInitFunc) camel_imap_store_init, + (CamelObjectFinalizeFunc) camel_imap_store_finalize); + } + + return type; +} + +static void +camel_imap_store_class_init (CamelIMAPStoreClass *klass) +{ + CamelServiceClass *service_class = (CamelServiceClass *) klass; + CamelStoreClass *store_class = (CamelStoreClass *) klass; + + parent_class = camel_type_get_global_classfuncs (CAMEL_STORE_TYPE); + + service_class->construct = imap_construct; + service_class->get_name = imap_get_name; + service_class->connect = imap_connect; + service_class->disconnect = imap_disconnect; + service_class->query_auth_types = imap_query_auth_types; + + store_class->get_folder = imap_get_folder; + store_class->create_folder = imap_create_folder; + store_class->delete_folder = imap_delete_folder; + store_class->rename_folder = imap_rename_folder; + store_class->sync = imap_sync; + store_class->get_folder_info = imap_get_folder_info; + store_class->subscribe_folder = imap_subscribe_folder; + store_class->unsubscribe_folder = imap_unsubscribe_folder; + store_class->noop = imap_noop; +} + +static void +camel_imap_store_init (CamelIMAPStore *store, CamelIMAPStoreClass *klass) +{ + store->engine = NULL; +} + +static void +camel_imap_store_finalize (CamelObject *object) +{ + CamelIMAPStore *store = (CamelIMAPStore *) object; + + if (store->engine) + camel_object_unref (store->engine); +} + + +static void +imap_construct (CamelService *service, CamelSession *session, CamelProvider *provider, CamelURL *url, CamelException *ex) +{ + CAMEL_SERVICE_CLASS (parent_class)->construct (service, session, provider, url, ex); +} + +static char * +imap_get_name (CamelService *service, gboolean brief) +{ + if (brief) + return g_strdup_printf (_("IMAP server %s"), service->url->host); + else + return g_strdup_printf (_("IMAP service for %s on %s"), + service->url->user, service->url->host); +} + +enum { + USE_SSL_NEVER, + USE_SSL_ALWAYS, + USE_SSL_WHEN_POSSIBLE +}; + +#define SSL_PORT_FLAGS (CAMEL_TCP_STREAM_SSL_ENABLE_SSL2 | CAMEL_TCP_STREAM_SSL_ENABLE_SSL3) +#define STARTTLS_FLAGS (CAMEL_TCP_STREAM_SSL_ENABLE_TLS) + +static gboolean +connect_to_server (CamelService *service, struct hostent *host, int ssl_mode, int try_starttls, CamelException *ex) +{ + CamelIMAPStore *store = (CamelIMAPStore *) service; + CamelIMAPEngine *engine; + CamelStream *tcp_stream; + int port, ret; + + if (store->engine) { + camel_object_unref (store->engine); + store->engine = NULL; + } + + port = service->url->port ? service->url->port : 143; + + if (ssl_mode) { +#ifdef HAVE_SSL + if (try_starttls) { + tcp_stream = camel_tcp_stream_ssl_new (service->session, service->url->host, STARTTLS_FLAGS); + } else { + port = service->url->port ? service->url->port : 993; + tcp_stream = camel_tcp_stream_ssl_new (service->session, service->url->host, SSL_PORT_FLAGS); + } +#else + if (!try_starttls) + port = service->url->port ? service->url->port : 993; + + camel_exception_setv (ex, CAMEL_EXCEPTION_SERVICE_UNAVAILABLE, + _("Could not connect to %s (port %d): %s"), + service->url->host, port, + _("SSL unavailable")); + + return FALSE; +#endif /* HAVE_SSL */ + } else { + tcp_stream = camel_tcp_stream_raw_new (); + } + + fprintf (stderr, "connecting to %s:%d\n", service->url->host, port); + if ((ret = camel_tcp_stream_connect ((CamelTcpStream *) tcp_stream, host, port)) == -1) { + if (errno == EINTR) + camel_exception_set (ex, CAMEL_EXCEPTION_USER_CANCEL, + _("Connection cancelled")); + else + camel_exception_setv (ex, CAMEL_EXCEPTION_SERVICE_UNAVAILABLE, + _("Could not connect to %s (port %d): %s"), + service->url->host, port, + g_strerror (errno)); + + camel_object_unref (tcp_stream); + + return FALSE; + } + + engine = camel_imap_engine_new (service->session, service->url); + if (camel_imap_engine_take_stream (engine, tcp_stream, ex) == -1) { + camel_object_unref (engine); + + return FALSE; + } + + if (camel_imap_engine_capability (engine, ex) == -1) { + camel_object_unref (engine); + + return FALSE; + } + + store->engine = engine; + +#ifdef HAVE_SSL + if (ssl_mode == USE_SSL_WHEN_POSSIBLE) { + /* try_starttls is always TRUE here */ + if (engine->capa & CAMEL_IMAP_CAPABILITY_STARTTLS) + goto starttls; + } else if (ssl_mode == USE_SSL_ALWAYS) { + if (try_starttls) { + if (engine->capa & CAMEL_IMAP_CAPABILITY_STARTTLS) { + goto starttls; + } else { + /* server doesn't support STARTTLS, abort */ + camel_exception_setv (ex, CAMEL_EXCEPTION_SERVICE_UNAVAILABLE, + _("Failed to connect to IMAP server %s in secure mode: " + "Server does not support STARTTLS"), + service->url->host); + goto exception; + } + } + } +#endif /* HAVE_SSL */ + + return TRUE; + +#ifdef HAVE_SSL + starttls: + + if (1) { + CamelIMAPCommand *ic; + int id; + + ic = camel_imap_engine_queue (engine, NULL, "STARTTLS\r\n"); + while ((id = camel_imap_engine_iterate (engine)) < ic->id && id != -1) + ; + + if (id == -1 || ic->result != CAMEL_IMAP_RESULT_OK) { + if (ic->result != CAMEL_IMAP_RESULT_OK) { + camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM, + _("Failed to connect to IMAP server %s in secure mode: %s"), + service->url->host, _("Unknown error")); + } else { + camel_exception_xfer (ex, &ic->ex); + } + + camel_imap_command_unref (ic); + + goto exception; + } + + camel_imap_command_unref (ic); + } + + return TRUE; + + exception: + + camel_object_unref (store->engine); + store->engine = NULL; + + return FALSE; +#endif /* HAVE_SSL */ +} + +static struct { + char *value; + int mode; +} ssl_options[] = { + { "", USE_SSL_ALWAYS }, + { "always", USE_SSL_ALWAYS }, + { "when-possible", USE_SSL_WHEN_POSSIBLE }, + { "never", USE_SSL_NEVER }, + { NULL, USE_SSL_NEVER }, +}; + +static gboolean +connect_to_server_wrapper (CamelService *service, CamelException *ex) +{ + const char *use_ssl; + struct hostent *h; + int ssl_mode; + int ret, i; + + if (!(h = camel_service_gethost (service, ex))) + return FALSE; + + if ((use_ssl = camel_url_get_param (service->url, "use_ssl"))) { + for (i = 0; ssl_options[i].value; i++) + if (!strcmp (ssl_options[i].value, use_ssl)) + break; + ssl_mode = ssl_options[i].mode; + } else { + ssl_mode = USE_SSL_NEVER; + } + + if (ssl_mode == USE_SSL_ALWAYS) { + /* First try the ssl port */ + if (!(ret = connect_to_server (service, h, ssl_mode, FALSE, err))) { + if (camel_exception_get_id (ex) == CAMEL_EXCEPTION_SERVICE_UNAVAILABLE) { + /* The ssl port seems to be unavailable, lets try STARTTLS */ + camel_exception_clear (ex); + ret = connect_to_server (service, h, ssl_mode, TRUE, ex); + } + } + } else if (ssl_mode == USE_SSL_WHEN_POSSIBLE) { + /* If the server supports STARTTLS, use it */ + ret = connect_to_server (service, h, ssl_mode, TRUE, ex); + } else { + /* User doesn't care about SSL */ + ret = connect_to_server (service, h, USE_SSL_ALWAYS, FALSE, err); + } + + camel_free_host (h); + + return ret; +} + +static int +sasl_auth (CamelIMAPEngine *engine, CamelIMAPCommand *ic, const unsigned char *linebuf, size_t linelen, CamelException *ex) +{ + /* Perform a single challenge iteration */ + CamelSasl *sasl = ic->user_data; + char *challenge; + + if (camel_sasl_authenticated (sasl)) { + camel_exception_setv (ex, CAMEL_EXCEPTION_SERVICE_CANT_AUTHENTICATE, + _("Cannot authenticate to IMAP server %s using the %s authentication mechanism"), + engine->url->host, engine->url->auth); + return -1; + } + + while (isspace (*linebuf)) + linebuf++; + + if (*linebuf == '\0') + linebuf = NULL; + + if (!(challenge = camel_sasl_challenge_base64 (sasl, (const char *) linebuf, err))) + return -1; + + fprintf (stderr, "sending : %s\r\n", challenge); + + if (camel_stream_printf (engine->ostream, "%s\r\n", challenge) == -1) { + g_free (challenge); + return -1; + } + + g_free (challenge); + + if (camel_stream_flush (engine->ostream) == -1) + return -1; + + return 0; +} + +static int +imap_try_authenticate (CamelService *service, gboolean reprompt, const char *errmsg, CamelException *ex) +{ + CamelIMAPStore *store = (CamelIMAPStore *) service; + CamelSession *session = service->session; + CamelSasl *sasl = NULL; + CamelIMAPCommand *ic; + int id; + + if (!service->url->passwd) { + guint32 flags = CAMEL_SESSION_PASSWORD_SECRET; + char *prompt; + + if (reprompt) + flags |= CAMEL_SESSION_PASSWORD_REPROMPT; + + prompt = g_strdup_printf (_("%sPlease enter the IMAP password for %s on host %s"), + errmsg ? errmsg : "", + service->url->user, + service->url->host); + + service->url->passwd = camel_session_get_password (session, prompt, flags, service, "password", ex); + + g_free (prompt); + + if (!service->url->passwd) + return FALSE; + } + + if (service->url->auth) { + CamelServiceAuthType *mech; + + mech = g_hash_table_lookup (store->engine->authtypes, service->url->auth); + sasl = camel_sasl_new ("imap", mech->authproto, service); + + ic = camel_imap_engine_queue (store->engine, NULL, "AUTHENTICATE %s\r\n", service->url->auth); + ic->plus = sasl_auth; + ic->user_data = sasl; + } else { + ic = camel_imap_engine_queue (store->engine, NULL, "LOGIN %S %S\r\n", + service->url->user, service->url->passwd); + } + + while ((id = camel_imap_engine_iterate (store->engine)) < ic->id && id != -1) + ; + + if (sasl != NULL) + camel_object_unref (sasl); + + if (id == -1 || ic->status == CAMEL_IMAP_COMMAND_ERROR) { + /* unrecoverable error */ + camel_exception_xfer (ex, &ic->ex); + camel_imap_command_unref (ic); + + return FALSE; + } + + if (ic->result != CAMEL_IMAP_RESULT_OK) { + camel_imap_command_unref (ic); + + /* try again */ + + return TRUE; + } + + camel_imap_command_unref (ic); + + return FALSE; +} + +static gboolean +imap_connect (CamelService *service, CamelException *ex) +{ + CamelIMAPStore *store = (CamelIMAPStore *) service; + CamelServiceAuthType *mech; + gboolean reprompt = FALSE; + char *errmsg = NULL; + CamelException lex; + + CAMEL_SERVICE_LOCK (store, connect_lock); + + if (!connect_to_server_wrapper (service, ex)) { + CAMEL_SERVICE_UNLOCK (store, connect_lock); + return FALSE; + } + +#define CANT_USE_AUTHMECH (!(mech = g_hash_table_lookup (store->engine->authtypes, service->url->auth))) + if (service->url->auth && CANT_USE_AUTHMECH) { + /* Oops. We can't AUTH using the requested mechanism */ + camel_exception_setv (ex, CAMEL_EXCEPTION_SERVICE_CANT_AUTHENTICATE, + _("Cannot authenticate to IMAP server %s using %s"), + service->url->host, service->url->auth); + + camel_object_unref (store->engine); + store->engine = NULL; + + CAMEL_SERVICE_UNLOCK (store, connect_lock); + + return FALSE; + } + + camel_exception_init (&lex); + while (imap_try_authenticate (service, reprompt, errmsg, &lex)) { + g_free (errmsg); + errmsg = g_strdup (lex.desc); + camel_exception_clear (&lex); + reprompt = TRUE; + } + g_free (errmsg); + + if (camel_exception_is_set (&lex)) { + camel_exception_xfer (ex, &lex); + camel_object_unref (store->engine); + store->engine = NULL; + + CAMEL_SERVICE_UNLOCK (store, connect_lock); + + return FALSE; + } + + if (camel_imap_engine_namespace (store->engine, ex) == -1) { + camel_object_unref (store->engine); + store->engine = NULL; + + CAMEL_SERVICE_UNLOCK (store, connect_lock); + + return FALSE; + } + + CAMEL_SERVICE_UNLOCK (store, connect_lock); + + return TRUE; +} + +static gboolean +imap_disconnect (CamelService *service, gboolean clean, CamelException *ex) +{ + CamelIMAPStore *store = (CamelIMAPStore *) service; + CamelIMAPCommand *ic; + int id; + + if (clean && !store->engine->istream->disconnected) { + ic = camel_imap_engine_queue (store->engine, NULL, "LOGOUT\r\n"); + while ((id = camel_imap_engine_iterate (store->engine)) < ic->id && id != -1) + ; + + camel_imap_command_unref (ic); + } + + camel_object_unref (store->engine); + + return 0; +} + +extern CamelServiceAuthType camel_imap_password_authtype; + +static GList * +imap_query_auth_types (CamelService *service, CamelException *ex) +{ + CamelIMAPStore *store = (CamelIMAPStore *) service; + CamelServiceAuthType *authtype; + GList *sasl_types, *t, *next; + gboolean connected; + + CAMEL_SERVICE_LOCK (store, connect_lock); + connected = connect_to_server_wrapper (service, ex); + CAMEL_SERVICE_UNLOCK (store, connect_lock); + if (!connected) + return NULL; + + sasl_types = camel_sasl_authtype_list (FALSE); + for (t = sasl_types; t; t = next) { + authtype = t->data; + next = t->next; + + if (!g_hash_table_lookup (store->engine->authtypes, authtype->authproto)) { + sasl_types = g_list_remove_link (sasl_types, t); + g_list_free_1 (t); + } + } + + return g_list_prepend (sasl_types, &camel_imap_password_authtype); +} + + +static char * +imap_folder_utf7_name (CamelStore *store, const char *folder_name) +{ + char *real_name, *p; + + if (store->dir_sep != '/') { + p = real_name = g_alloca (strlen (folder_name) + 1); + strcpy (real_name, folder_name); + while (*p != '\0') { + if (*p == '/') + *p = store->dir_sep; + p++; + } + + folder_name = real_name; + } + + return camel_utf8_utf7 (folder_name); +} + +static CamelFolder * +imap_get_folder (CamelStore *store, const char *folder_name, guint32 flags, CamelException *ex) +{ + CamelIMAPStore *imap_store = (CamelIMAPStore *) store; + + /* FIXME: implement me */ + + return NULL; +} + +static CamelFolderInfo * +imap_create_folder (CamelStore *store, const char *parent_name, const char *folder_name, CamelException *ex) +{ + /* FIXME: also need to deal with parent folders that can't + * contain subfolders - delete them and re-create with the + * proper hint */ + CamelIMAPEngine *engine = ((CamelIMAPStore *) folder->store)->engine; + CamelFolderInfo *fi = NULL + CamelIMAPCommand *ic; + char *utf7_name; + const char *c; + char *name; + int id; + + c = folder_name; + while (*c != '\0') { + if (*c == store->dir_sep || strchr ("#%*", *c)) { + camel_exception_setv (ex, CAMEL_EXCEPTION_FOLDER_INVALID_PATH, + _("The folder name \"%s\" is invalid because " + "it containes the character \"%c\""), + folder_name, *c); + return NULL; + } + + c++; + } + + if (parent_name != NULL && *parent_name) + name = g_strdup_printf ("%s/%s", parent_name, folder_name); + else + name = g_strdup (folder_name); + + utf7_name = imap_folder_utf7_name (store, name); + g_free (name); + + ic = camel_imap_engine_queue (engine, NULL, "CREATE %S\r\n", utf7_name); + g_free (utf7_name); + + while ((id = camel_imap_engine_iterate (engine)) < ic->id && id != -1) + ; + + if (id == -1 || ic->status != CAMEL_IMAP_COMMAND_COMPLETE) { + camel_exception_xfer (ex, &ic->ex); + camel_imap_command_unref (ic); + return NULL; + } + + switch (ic->result) { + case CAMEL_IMAP_RESULT_OK: + /* FIXME: allocate fi */ + break; + case CAMEL_IMAP_RESULT_NO: + /* FIXME: would be good to save the NO reason into the err message */ + camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM, + _("Cannot create folder `%s': Invalid mailbox name"), + folder_name); + break; + case CAMEL_IMAP_RESULT_BAD: + camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM, + _("Cannot create folder `%s': Bad command"), + folder_name); + break; + default: + g_assert_not_reached (); + } + + camel_imap_command_unref (ic); + + return fi; +} + +static void +imap_delete_folder (CamelStore *store, const char *folder_name, CamelException *ex) +{ + CamelIMAPEngine *engine = ((CamelIMAPStore *) folder->store)->engine; + CamelIMAPCommand *ic; + char *utf7_name; + int id; + + if (!g_ascii_strcasecmp (folder_name, "INBOX")) { + camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM, + _("Cannot delete folder `%s': Special folder"), + folder_name); + + return; + } + + utf7_name = imap_folder_utf7_name (store, folder_name); + ic = camel_imap_engine_queue (engine, NULL, "DELETE %S\r\n", utf7_name); + g_free (utf7_name); + + while ((id = camel_imap_engine_iterate (engine)) < ic->id && id != -1) + ; + + if (id == -1 || ic->status != CAMEL_IMAP_COMMAND_COMPLETE) { + camel_exception_xfer (ex, &ic->ex); + camel_imap_command_unref (ic); + return; + } + + switch (ic->result) { + case CAMEL_IMAP_RESULT_OK: + /* deleted */ + /*fi = imap_build_folder_info (store, folder_name); + camel_object_trigger_event (store, "folder_deleted", fi); + camel_folder_info_free (fi);*/ + break; + case CAMEL_IMAP_RESULT_NO: + /* FIXME: would be good to save the NO reason into the err message */ + camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM, + _("Cannot delete folder `%s': Invalid mailbox name"), + folder_name); + break; + case CAMEL_IMAP_RESULT_BAD: + camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM, + _("Cannot delete folder `%s': Bad command"), + folder_name); + break; + } + + camel_imap_command_unref (ic); +} + +static void +imap_rename_folder (CamelStore *store, const char *old_name, const char *new_name, CamelException *ex) +{ + +} + +static void +imap_sync (CamelStore *store, gboolean expunge, CamelException *ex) +{ + +} + +static CamelFolderInfo * +imap_get_folder_info (CamelStore *store, const char *top, guint32 flags, CamelException *ex) +{ + +} + +static void +imap_subscribe_folder (CamelStore *store, const char *folder_name, CamelException *ex) +{ + CamelIMAPEngine *engine = ((CamelIMAPStore *) store)->engine; + CamelIMAPCommand *ic; + char *utf7_name; + int id; + + utf7_name = imap_fodler_utf7_name (store, folder_name); + ic = camel_imap_engine_queue (engine, NULL, "SUBSCRIBE %S\r\n", utf7_name); + g_free (utf7_name); + + while ((id = camel_imap_engine_iterate (engine)) < ic->id && id != -1) + ; + + if (id == -1 || ic->status != CAMEL_IMAP_COMMAND_COMPLETE) { + camel_exception_xfer (ex, &ic->ex); + camel_imap_command_unref (ic); + return; + } + + switch (ic->result) { + case CAMEL_IMAP_RESULT_OK: + /* subscribed */ + /*fi = imap_build_folder_info (store, folder_name); + fi->flags |= CAMEL_FOLDER_NOCHILDREN; + camel_object_trigger_event (store, "folder_subscribed", fi); + camel_folder_info_free (fi);*/ + break; + case CAMEL_IMAP_RESULT_NO: + /* FIXME: would be good to save the NO reason into the err message */ + camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM, + _("Cannot subscribe to folder `%s': Invalid mailbox name"), + folder_name); + break; + case CAMEL_IMAP_RESULT_BAD: + camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM, + _("Cannot subscribe to folder `%s': Bad command"), + folder_name); + break; + } + + camel_imap_command_unref (ic); +} + +static void +imap_unsubscribe_folder (CamelStore *store, const char *folder_name, CamelException *ex) +{ + CamelIMAPEngine *engine = ((CamelIMAPStore *) store)->engine; + CamelIMAPCommand *ic; + char *utf7_name; + int id; + + utf7_name = imap_fodler_utf7_name (store, folder_name); + ic = camel_imap_engine_queue (engine, NULL, "UNSUBSCRIBE %S\r\n", utf7_name); + g_free (utf7_name); + + while ((id = camel_imap_engine_iterate (engine)) < ic->id && id != -1) + ; + + if (id == -1 || ic->status != CAMEL_IMAP_COMMAND_COMPLETE) { + camel_exception_xfer (ex, &ic->ex); + camel_imap_command_unref (ic); + return; + } + + switch (ic->result) { + case CAMEL_IMAP_RESULT_OK: + /* unsubscribed */ + /*fi = imap_build_folder_info (store, folder_name); + camel_object_trigger_event (store, "folder_unsubscribed", fi); + camel_folder_info_free (fi);*/ + break; + case CAMEL_IMAP_RESULT_NO: + /* FIXME: would be good to save the NO reason into the err message */ + camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM, + _("Cannot unsubscribe from folder `%s': Invalid mailbox name"), + folder_name); + break; + case CAMEL_IMAP_RESULT_BAD: + camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM, + _("Cannot unsubscribe from folder `%s': Bad command"), + folder_name); + break; + } + + camel_imap_command_unref (ic); +} + +static void +imap_noop (CamelStore *store, CamelException *ex) +{ + +} diff --git a/camel/providers/imap4/camel-imap-store.h b/camel/providers/imap4/camel-imap-store.h new file mode 100644 index 0000000000..036c9808a9 --- /dev/null +++ b/camel/providers/imap4/camel-imap-store.h @@ -0,0 +1,61 @@ +/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ +/* Camel + * Copyright (C) 1999-2004 Jeffrey Stedfast + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Street #330, Boston, MA 02111-1307, USA. + */ + + +#ifndef __CAMEL_IMAP_STORE_H__ +#define __CAMEL_IMAP_STORE_H__ + +#include <camel/camel-store.h> + +#ifdef __cplusplus +extern "C" { +#pragma } +#endif /* __cplusplus */ + +#define CAMEL_TYPE_IMAP_STORE (camel_imap_store_get_type ()) +#define CAMEL_IMAP_STORE(obj) (CAMEL_CHECK_CAST ((obj), CAMEL_TYPE_IMAP_STORE, CamelIMAPStore)) +#define CAMEL_IMAP_STORE_CLASS(klass) (CAMEL_CHECK_CLASS_CAST ((klass), CAMEL_TYPE_IMAP_STORE, CamelIMAPStoreClass)) +#define CAMEL_IS_IMAP_STORE(obj) (CAMEL_CHECK_TYPE ((obj), CAMEL_TYPE_IMAP_STORE)) +#define CAMEL_IS_IMAP_STORE_CLASS(klass) (CAMEL_CHECK_CLASS_TYPE ((klass), CAMEL_TYPE_IMAP_STORE)) +#define CAMEL_IMAP_STORE_GET_CLASS(obj) (CAMEL_CHECK_GET_CLASS ((obj), CAMEL_TYPE_IMAP_STORE, CamelIMAPStoreClass)) + +typedef struct _CamelIMAPStore CamelIMAPStore; +typedef struct _CamelIMAPStoreClass CamelIMAPStoreClass; + +struct _CamelIMAPEngine; + +struct _CamelIMAPStore { + CamelStore parent_object; + + struct _CamelIMAPEngine *engine; +}; + +struct _CamelIMAPStoreClass { + CamelStoreClass parent_class; + +}; + + +CamelType spruce_imap_store_get_type (void); + +#ifdef __cplusplus +} +#endif /* __cplusplus */ + +#endif /* __CAMEL_IMAP_STORE_H__ */ |