X-Git-Url: https://git.notmuchmail.org/git?p=notmuch;a=blobdiff_plain;f=message.cc;h=7a6c01d2b7ace590b9fde3a93b90d65c8f934a06;hp=65da4b878fa65d03efd07b732e04eec5d17ee708;hb=6b20dbff86c2be180496b8ed762b2b68b10a772a;hpb=28dd86af05688e96f7da8297099230b77d34f17a diff --git a/message.cc b/message.cc index 65da4b87..7a6c01d2 100644 --- a/message.cc +++ b/message.cc @@ -42,8 +42,7 @@ struct _notmuch_tags { }; struct _notmuch_thread_ids { - char *current; - char *next; + notmuch_terms_t terms; }; /* "128 bits of thread-id ought to be enough for anybody" */ @@ -81,23 +80,43 @@ _notmuch_message_destructor (notmuch_message_t *message) * caller *is* responsible for calling notmuch_message_destroy. * * If no document exists in the database with document ID of 'doc_id' - * then this function returns NULL. + * then this function returns NULL and sets *status to + * NOTMUCH_PRIVATE_STATUS_NO_DOCUMENT_FOUND. + * + * This function can also fail to due lack of available memory, + * returning NULL and optionally setting *status to + * NOTMUCH_PRIVATE_STATUS_OUT_OF_MEMORY. Caller can pass NULL + * for status if uninterested in distinguishing these two cases. */ notmuch_message_t * _notmuch_message_create (const void *talloc_owner, notmuch_database_t *notmuch, - unsigned int doc_id) + unsigned int doc_id, + notmuch_private_status_t *status) { notmuch_message_t *message; + if (status) + *status = NOTMUCH_PRIVATE_STATUS_SUCCESS; + message = talloc (talloc_owner, notmuch_message_t); - if (unlikely (message == NULL)) + if (unlikely (message == NULL)) { + if (status) + *status = NOTMUCH_PRIVATE_STATUS_OUT_OF_MEMORY; return NULL; + } message->notmuch = notmuch; message->doc_id = doc_id; message->message_id = NULL; /* lazily created */ message->filename = NULL; /* lazily created */ + + /* This is C++'s creepy "placement new", which is really just an + * ugly way to call a constructor for a pre-allocated object. So + * it's really not an error to not be checking for OUT_OF_MEMORY + * here, since this "new" isn't actually allocating memory. This + * is language-design comedy of the wrong kind. */ + new (&message->doc) Xapian::Document; talloc_set_destructor (message, _notmuch_message_destructor); @@ -106,6 +125,8 @@ _notmuch_message_create (const void *talloc_owner, message->doc = notmuch->xapian_db->get_document (doc_id); } catch (const Xapian::DocNotFoundError &error) { talloc_free (message); + if (status) + *status = NOTMUCH_PRIVATE_STATUS_NO_DOCUMENT_FOUND; return NULL; } @@ -128,32 +149,60 @@ _notmuch_message_create (const void *talloc_owner, * If there is already a document with message ID 'message_id' in the * database, then the returned message can be used to query/modify the * document. Otherwise, a new document will be inserted into the - * database before this function returns; + * database before this function returns. + * + * If an error occurs, this function will return NULL and *status + * will be set as appropriate. (The status pointer argument must + * not be NULL.) */ notmuch_message_t * _notmuch_message_create_for_message_id (const void *talloc_owner, notmuch_database_t *notmuch, - const char *message_id) + const char *message_id, + notmuch_status_t *status) { + notmuch_private_status_t private_status; notmuch_message_t *message; Xapian::Document doc; unsigned int doc_id; char *term; + *status = NOTMUCH_STATUS_SUCCESS; + message = notmuch_database_find_message (notmuch, message_id); if (message) return talloc_steal (talloc_owner, message); term = talloc_asprintf (NULL, "%s%s", _find_prefix ("id"), message_id); - doc.add_term (term); - talloc_free (term); + if (term == NULL) { + *status = NOTMUCH_STATUS_OUT_OF_MEMORY; + return NULL; + } + + try { + doc.add_term (term); + talloc_free (term); + + doc.add_value (NOTMUCH_VALUE_MESSAGE_ID, message_id); + + doc_id = notmuch->xapian_db->add_document (doc); + } catch (const Xapian::Error &error) { + *status = NOTMUCH_STATUS_XAPIAN_EXCEPTION; + return NULL; + } - doc.add_value (NOTMUCH_VALUE_MESSAGE_ID, message_id); + message = _notmuch_message_create (talloc_owner, notmuch, + doc_id, &private_status); - doc_id = notmuch->xapian_db->add_document (doc); + if (private_status >= (notmuch_private_status_t) NOTMUCH_STATUS_LAST_STATUS) + { + INTERNAL_ERROR ("Failed to find document immediately after adding it.\n"); + } + + *status = (notmuch_status_t) private_status; - return _notmuch_message_create (talloc_owner, notmuch, doc_id); + return message; } const char * @@ -168,9 +217,8 @@ notmuch_message_get_message_id (notmuch_message_t *message) i.skip_to (_find_prefix ("id")); if (i == message->doc.termlist_end ()) { - fprintf (stderr, "Internal error: Message with document ID of %d has no message ID.\n", - message->doc_id); - exit (1); + INTERNAL_ERROR ("Message with document ID of %d has no message ID.\n", + message->doc_id); } message->message_id = talloc_strdup (message, (*i).c_str () + 1); @@ -255,6 +303,10 @@ _notmuch_terms_create (void *ctx, return terms; } +/* The assertion is to ensure that 'type' is a derivative of + * notmuch_terms_t in that it contains a notmuch_terms_t as its first + * member. We do this by name of 'terms' as opposed to type, because + * that's as clever as I've been so far. */ #define _notmuch_terms_create_type(ctx, doc, prefix_name, type) \ (COMPILE_TIME_ASSERT(offsetof(type, terms) == 0), \ (type *) _notmuch_terms_create (ctx, doc, prefix_name)) @@ -269,20 +321,8 @@ notmuch_message_get_tags (notmuch_message_t *message) notmuch_thread_ids_t * notmuch_message_get_thread_ids (notmuch_message_t *message) { - notmuch_thread_ids_t *thread_ids; - std::string id_str; - - thread_ids = talloc (message, notmuch_thread_ids_t); - if (unlikely (thread_ids == NULL)) - return NULL; - - id_str = message->doc.get_value (NOTMUCH_VALUE_THREAD); - thread_ids->next = talloc_strdup (message, id_str.c_str ()); - - /* Initialize thread_ids->current and terminate first ID. */ - notmuch_thread_ids_advance (thread_ids); - - return thread_ids; + return _notmuch_terms_create_type (message, message->doc, "thread", + notmuch_thread_ids_t); } void @@ -301,26 +341,7 @@ void _notmuch_message_add_thread_id (notmuch_message_t *message, const char *thread_id) { - std::string id_str; - _notmuch_message_add_term (message, "thread", thread_id); - - id_str = message->doc.get_value (NOTMUCH_VALUE_THREAD); - - if (id_str.empty ()) { - message->doc.add_value (NOTMUCH_VALUE_THREAD, thread_id); - } else { - size_t pos; - - /* Think about using a hash here if there's any performance - * problem. */ - pos = id_str.find (thread_id); - if (pos == std::string::npos) { - id_str.append (","); - id_str.append (thread_id); - message->doc.add_value (NOTMUCH_VALUE_THREAD, id_str); - } - } } static void @@ -360,7 +381,6 @@ _notmuch_message_ensure_thread_id (notmuch_message_t *message) thread_id_generate (&thread_id); _notmuch_message_add_term (message, "thread", thread_id.str); - message->doc.add_value (NOTMUCH_VALUE_THREAD, thread_id.str); } /* Synchronize changes made to message->doc out into the database. */ @@ -446,9 +466,8 @@ notmuch_message_add_tag (notmuch_message_t *message, const char *tag) status = _notmuch_message_add_term (message, "tag", tag); if (status) { - fprintf (stderr, "Internal error: _notmuch_message_add_term return unexpected value: %d\n", - status); - exit (1); + INTERNAL_ERROR ("_notmuch_message_add_term return unexpected value: %d\n", + status); } _notmuch_message_sync (message); @@ -469,9 +488,8 @@ notmuch_message_remove_tag (notmuch_message_t *message, const char *tag) status = _notmuch_message_remove_term (message, "tag", tag); if (status) { - fprintf (stderr, "Internal error: _notmuch_message_remove_term return unexpected value: %d\n", - status); - exit (1); + INTERNAL_ERROR ("_notmuch_message_remove_term return unexpected value: %d\n", + status); } _notmuch_message_sync (message); @@ -545,26 +563,23 @@ notmuch_tags_destroy (notmuch_tags_t *tags) notmuch_bool_t notmuch_thread_ids_has_more (notmuch_thread_ids_t *thread_ids) { - if (thread_ids->current == NULL || *thread_ids->current == '\0') - return FALSE; - else - return TRUE; + return _notmuch_terms_has_more (&thread_ids->terms); } const char * notmuch_thread_ids_get (notmuch_thread_ids_t *thread_ids) { - return thread_ids->current; + return _notmuch_terms_get (&thread_ids->terms); } void notmuch_thread_ids_advance (notmuch_thread_ids_t *thread_ids) { - thread_ids->current = strsep (&thread_ids->next, ","); + return _notmuch_terms_advance (&thread_ids->terms); } void notmuch_thread_ids_destroy (notmuch_thread_ids_t *thread_ids) { - talloc_free (thread_ids); + return _notmuch_terms_destroy (&thread_ids->terms); }