aboutsummaryrefslogtreecommitdiffstats
path: root/camel/providers
diff options
context:
space:
mode:
Diffstat (limited to 'camel/providers')
-rw-r--r--camel/providers/pop3/camel-pop3-folder.c187
1 files changed, 119 insertions, 68 deletions
diff --git a/camel/providers/pop3/camel-pop3-folder.c b/camel/providers/pop3/camel-pop3-folder.c
index 0032077fd6..2648a48a5b 100644
--- a/camel/providers/pop3/camel-pop3-folder.c
+++ b/camel/providers/pop3/camel-pop3-folder.c
@@ -35,6 +35,8 @@
#include "camel-mime-message.h"
#include "camel-operation.h"
+#include <e-util/md5-utils.h>
+
#include <stdlib.h>
#include <string.h>
@@ -49,6 +51,8 @@ static void pop3_sync (CamelFolder *folder, gboolean expunge,
static gint pop3_get_message_count (CamelFolder *folder);
static GPtrArray *pop3_get_uids (CamelFolder *folder);
+static CamelStreamMem *pop3_get_message_stream (CamelFolder *folder, int id,
+ CamelException *ex);
static CamelMimeMessage *pop3_get_message (CamelFolder *folder,
const char *uid,
CamelException *ex);
@@ -62,17 +66,17 @@ camel_pop3_folder_class_init (CamelPop3FolderClass *camel_pop3_folder_class)
{
CamelFolderClass *camel_folder_class =
CAMEL_FOLDER_CLASS (camel_pop3_folder_class);
-
+
parent_class = CAMEL_FOLDER_CLASS(camel_type_get_global_classfuncs (camel_folder_get_type ()));
-
+
/* virtual method overload */
camel_folder_class->refresh_info = pop3_refresh_info;
camel_folder_class->sync = pop3_sync;
-
+
camel_folder_class->get_message_count = pop3_get_message_count;
camel_folder_class->get_uids = pop3_get_uids;
camel_folder_class->free_uids = camel_folder_free_nop;
-
+
camel_folder_class->get_message = pop3_get_message;
camel_folder_class->set_message_flags = pop3_set_message_flags;
}
@@ -82,10 +86,10 @@ camel_pop3_folder_init (gpointer object)
{
CamelFolder *folder = CAMEL_FOLDER (object);
CamelPop3Folder *pop3_folder = CAMEL_POP3_FOLDER (object);
-
+
folder->has_summary_capability = FALSE;
folder->has_search_capability = FALSE;
-
+
pop3_folder->uids = NULL;
pop3_folder->flags = NULL;
}
@@ -94,7 +98,7 @@ CamelType
camel_pop3_folder_get_type (void)
{
static CamelType camel_pop3_folder_type = CAMEL_INVALID_TYPE;
-
+
if (!camel_pop3_folder_type) {
camel_pop3_folder_type = camel_type_register (CAMEL_FOLDER_TYPE, "CamelPop3Folder",
sizeof (CamelPop3Folder),
@@ -104,7 +108,7 @@ camel_pop3_folder_get_type (void)
(CamelObjectInitFunc) camel_pop3_folder_init,
(CamelObjectFinalizeFunc) pop3_finalize);
}
-
+
return camel_pop3_folder_type;
}
@@ -112,7 +116,7 @@ void
pop3_finalize (CamelObject *object)
{
CamelPop3Folder *pop3_folder = CAMEL_POP3_FOLDER (object);
-
+
if (pop3_folder->uids)
camel_folder_free_deep (NULL, pop3_folder->uids);
if (pop3_folder->flags)
@@ -123,31 +127,71 @@ CamelFolder *
camel_pop3_folder_new (CamelStore *parent, CamelException *ex)
{
CamelFolder *folder;
-
+
folder = CAMEL_FOLDER (camel_object_new (CAMEL_POP3_FOLDER_TYPE));
camel_folder_construct (folder, parent, "inbox", "inbox");
-
+
/* mt-ok, since we dont have the folder-lock for new() */
camel_folder_refresh_info (folder, ex);/* mt-ok */
if (camel_exception_is_set (ex)) {
camel_object_unref (CAMEL_OBJECT (folder));
folder = NULL;
}
-
+
return folder;
}
+static GPtrArray *
+pop3_generate_uids (CamelFolder *folder, int count, CamelException *ex)
+{
+ /* FIXME: This is extremely inefficient but there's no way
+ around it. Because of that, perhaps we should have some way
+ of caching these messages so we don't have to download them
+ a second time for this session? */
+ GPtrArray *uids;
+ int i;
+
+ uids = g_ptr_array_new ();
+ g_ptr_array_set_size (uids, count);
+
+ for (i = 0; i < count; i++) {
+ CamelStreamMem *stream;
+ guchar digest[16];
+ char *uid;
+
+ stream = pop3_get_message_stream (folder, i, ex);
+ if (stream == NULL)
+ goto exception;
+
+ md5_get_digest (stream->buffer->data, stream->buffer->len, digest);
+ camel_object_unref (CAMEL_OBJECT (stream));
+
+ uid = base64_encode_simple (digest, 16);
+ g_ptr_array_add (uids, uid);
+ }
+
+ return uids;
+
+ exception:
+
+ for (i = 0; i < count; i++)
+ g_free (uids->pdata[i]);
+ g_ptr_array_free (uids, TRUE);
+
+ return NULL;
+}
+
static void
pop3_refresh_info (CamelFolder *folder, CamelException *ex)
{
+ CamelPop3Store *pop3_store = CAMEL_POP3_STORE (folder->parent_store);
+ CamelPop3Folder *pop3_folder = (CamelPop3Folder *) folder;
GPtrArray *uids;
int status, count;
char *data;
- CamelPop3Folder *pop3_folder = (CamelPop3Folder *) folder;
- CamelPop3Store *pop3_store = CAMEL_POP3_STORE (folder->parent_store);
-
- camel_operation_start(NULL, _("Retrieving POP summary"));
-
+
+ camel_operation_start (NULL, _("Retrieving POP summary"));
+
status = camel_pop3_command (pop3_store, &data, ex, "STAT");
switch (status) {
case CAMEL_POP3_ERR:
@@ -157,13 +201,13 @@ pop3_refresh_info (CamelFolder *folder, CamelException *ex)
g_free (data);
/* fall through */
case CAMEL_POP3_FAIL:
- camel_operation_end(NULL);
+ camel_operation_end (NULL);
return;
}
-
+
count = atoi (data);
g_free (data);
-
+
if (pop3_store->supports_uidl != FALSE) {
status = camel_pop3_command (pop3_store, NULL, ex, "UIDL");
switch (status) {
@@ -171,29 +215,24 @@ pop3_refresh_info (CamelFolder *folder, CamelException *ex)
pop3_store->supports_uidl = FALSE;
break;
case CAMEL_POP3_FAIL:
- camel_operation_end(NULL);
+ camel_operation_end (NULL);
return;
}
}
-
+
if (pop3_store->supports_uidl == FALSE) {
- int i;
-
- uids = g_ptr_array_new ();
- g_ptr_array_set_size (uids, count);
-
- for (i = 0; i < count; i++)
- uids->pdata[i] = g_strdup_printf ("%d", i + 1);
- camel_operation_end(NULL);
+ uids = pop3_generate_uids (folder, count, ex);
+
+ camel_operation_end (NULL);
} else {
data = camel_pop3_command_get_additional_data (pop3_store, 0, ex);
camel_operation_end(NULL);
if (camel_exception_is_set (ex))
return;
-
+
uids = parse_listing (count, data);
g_free (data);
-
+
if (!uids) {
camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
_("Could not open folder: "
@@ -202,7 +241,7 @@ pop3_refresh_info (CamelFolder *folder, CamelException *ex)
return;
}
}
-
+
pop3_folder->uids = uids;
pop3_folder->flags = g_new0 (guint32, uids->len);
}
@@ -216,10 +255,10 @@ pop3_sync (CamelFolder *folder, gboolean expunge, CamelException *ex)
if (!expunge)
return;
-
+
pop3_folder = CAMEL_POP3_FOLDER (folder);
pop3_store = CAMEL_POP3_STORE (folder->parent_store);
-
+
for (i = 0; i < pop3_folder->uids->len; i++) {
if (pop3_folder->flags[i] & CAMEL_MESSAGE_DELETED) {
status = camel_pop3_command (pop3_store, NULL, ex,
@@ -228,7 +267,7 @@ pop3_sync (CamelFolder *folder, gboolean expunge, CamelException *ex)
return;
}
}
-
+
camel_pop3_store_expunge (pop3_store, ex);
}
@@ -239,10 +278,10 @@ parse_listing (int count, char *data)
GPtrArray *ans;
char *p;
int index, len;
-
+
ans = g_ptr_array_new ();
g_ptr_array_set_size (ans, count);
-
+
p = data;
while (*p) {
index = strtoul (p, &p, 10);
@@ -253,14 +292,14 @@ parse_listing (int count, char *data)
if (*p == '\n')
p++;
}
-
+
for (index = 0; index < count; index++) {
if (ans->pdata[index] == NULL) {
g_ptr_array_free (ans, TRUE);
return NULL;
}
}
-
+
return ans;
}
@@ -268,35 +307,26 @@ static int
uid_to_number (CamelPop3Folder *pop3_folder, const char *uid)
{
int i;
-
+
for (i = 0; i < pop3_folder->uids->len; i++) {
if (!strcmp (uid, pop3_folder->uids->pdata[i]))
return i + 1;
}
-
+
return -1;
}
-
-static CamelMimeMessage *
-pop3_get_message (CamelFolder *folder, const char *uid, CamelException *ex)
+static CamelStreamMem *
+pop3_get_message_stream (CamelFolder *folder, int id, CamelException *ex)
{
- int status, num, total;
+ CamelStream *stream;
char *result, *body;
- CamelStream *msgstream;
- CamelMimeMessage *msg;
+ int status, total;
- num = uid_to_number (CAMEL_POP3_FOLDER (folder), uid);
- if (num == -1) {
- camel_exception_setv (ex, CAMEL_EXCEPTION_FOLDER_INVALID_UID,
- _("No message with uid %s"), uid);
- return NULL;
- }
-
- camel_operation_start_transient (NULL, _("Retrieving POP message %d"), num);
+ camel_operation_start_transient (NULL, _("Retrieving POP message %d"), id);
status = camel_pop3_command (CAMEL_POP3_STORE (folder->parent_store),
- &result, ex, "RETR %d", num);
+ &result, ex, "RETR %d", id);
switch (status) {
case CAMEL_POP3_ERR:
camel_exception_setv (ex, CAMEL_EXCEPTION_SERVICE_UNAVAILABLE,
@@ -323,18 +353,39 @@ pop3_get_message (CamelFolder *folder, const char *uid, CamelException *ex)
return NULL;
}
- msgstream = camel_stream_mem_new_with_buffer (body, strlen (body));
+ stream = camel_stream_mem_new_with_buffer (body, strlen (body));
g_free (body);
- msg = camel_mime_message_new ();
- camel_data_wrapper_construct_from_stream (CAMEL_DATA_WRAPPER (msg),
- CAMEL_STREAM (msgstream));
+ camel_operation_end (NULL);
- camel_object_unref (CAMEL_OBJECT (msgstream));
+ return CAMEL_STREAM_MEM (stream);
+}
+
+static CamelMimeMessage *
+pop3_get_message (CamelFolder *folder, const char *uid, CamelException *ex)
+{
+ CamelMimeMessage *message;
+ CamelStreamMem *stream;
+ int id;
- camel_operation_end (NULL);
+ id = uid_to_number (CAMEL_POP3_FOLDER (folder), uid);
+ if (id == -1) {
+ camel_exception_setv (ex, CAMEL_EXCEPTION_FOLDER_INVALID_UID,
+ _("No message with uid %s"), uid);
+ return NULL;
+ }
+
+ stream = pop3_get_message_stream (folder, id, ex);
+ if (stream == NULL)
+ return NULL;
+
+ message = camel_mime_message_new ();
+ camel_data_wrapper_construct_from_stream (CAMEL_DATA_WRAPPER (message),
+ CAMEL_STREAM (stream));
- return msg;
+ camel_object_unref (CAMEL_OBJECT (stream));
+
+ return message;
}
static void
@@ -343,11 +394,11 @@ pop3_set_message_flags (CamelFolder *folder, const char *uid,
{
CamelPop3Folder *pop3_folder = CAMEL_POP3_FOLDER (folder);
int num;
-
+
num = uid_to_number (pop3_folder, uid);
if (num == -1)
return;
-
+
pop3_folder->flags[num - 1] =
(pop3_folder->flags[num] & ~flags) | (set & flags);
}
@@ -356,7 +407,7 @@ static gint
pop3_get_message_count (CamelFolder *folder)
{
CamelPop3Folder *pop3_folder = CAMEL_POP3_FOLDER (folder);
-
+
return pop3_folder->uids->len;
}
@@ -364,6 +415,6 @@ static GPtrArray *
pop3_get_uids (CamelFolder *folder)
{
CamelPop3Folder *pop3_folder = CAMEL_POP3_FOLDER (folder);
-
+
return pop3_folder->uids;
}