]> git.ipfire.org Git - thirdparty/dovecot/core.git/commitdiff
lib-storage: Add mail_search_seqset_iter_*() API for batched iteration of a seqset
authorsergey.kitov <sergey.kitov@open-xchange.com>
Tue, 8 Mar 2022 09:52:31 +0000 (11:52 +0200)
committertimo.sirainen <timo.sirainen@open-xchange.com>
Mon, 28 Mar 2022 15:41:32 +0000 (15:41 +0000)
src/lib-storage/mail-search.c
src/lib-storage/mail-search.h

index 1b14b67d48d433b8f287d4df7299187756b56457..2488219f642592ee2296ff9ecdb5906704f5cb60 100644 (file)
@@ -804,3 +804,86 @@ void mail_search_args_result_deserialize(struct mail_search_args *args,
        for (arg = args->args; arg != NULL; arg = arg->next)
                mail_search_args_result_deserialize_arg(arg, &data, &size);
 }
+
+struct mail_search_seqset_iter {
+       struct mail_search_args *search_args;
+       ARRAY_TYPE(seq_range) seqset_left;
+       unsigned int batch_size;
+};
+
+static void mail_search_seqset_next_batch(struct mail_search_seqset_iter *iter)
+{
+       array_clear(&iter->search_args->args->value.seqset);
+       seq_range_array_merge_n(&iter->search_args->args->value.seqset,
+                               &iter->seqset_left, iter->batch_size);
+}
+
+struct mail_search_seqset_iter *
+mail_search_seqset_iter_init(struct mail_search_args *search_args,
+                            uint32_t messages_count, unsigned int batch_size)
+{
+       struct mail_search_seqset_iter *iter;
+
+       i_assert(search_args->args->next == NULL);
+
+       iter = i_new(struct mail_search_seqset_iter, 1);
+       iter->search_args = search_args;
+       iter->batch_size = batch_size;
+       mail_search_args_ref(iter->search_args);
+
+       switch (search_args->args->type) {
+       case SEARCH_SEQSET:
+       case SEARCH_UIDSET:
+               break;
+       case SEARCH_ALL:
+               if (search_args->args->match_not) {
+                       /* $ used before search result was saved */
+                       return iter;
+               }
+               /* 1:* - convert to seqset */
+               search_args->args->type = SEARCH_SEQSET;
+               p_array_init(&search_args->args->value.seqset,
+                            search_args->pool, 1);
+               seq_range_array_add_range(&search_args->args->value.seqset,
+                                         1, messages_count);
+               break;
+       default:
+               i_panic("Unexpected search_args type %d",
+                       search_args->args->type);
+       }
+
+       i_assert(search_args->args->type == SEARCH_SEQSET ||
+                search_args->args->type == SEARCH_UIDSET);
+
+       i_array_init(&iter->seqset_left,
+                    array_count(&search_args->args->value.seqset));
+       array_append_array(&iter->seqset_left, &search_args->args->value.seqset);
+       mail_search_seqset_next_batch(iter);
+       return iter;
+}
+
+void mail_search_seqset_iter_deinit(struct mail_search_seqset_iter **_iter)
+{
+       struct mail_search_seqset_iter *iter = *_iter;
+
+       if (iter == NULL)
+               return;
+
+       mail_search_args_unref(&iter->search_args);
+       array_free(&iter->seqset_left);
+       i_free(iter);
+}
+
+bool mail_search_seqset_iter_next(struct mail_search_seqset_iter *iter)
+{
+       if (!array_is_created(&iter->seqset_left) ||
+           array_count(&iter->seqset_left) == 0)
+               return FALSE;
+
+       /* remove the last batch of searched mails from seqset_left */
+       if (seq_range_array_remove_seq_range(&iter->seqset_left,
+                       &iter->search_args->args->value.seqset) == 0)
+               i_unreached();
+       mail_search_seqset_next_batch(iter);
+       return array_count(&iter->search_args->args->value.seqset) > 0;
+}
index fb46111bef409ec302abfd180a7b04d1d91ad247..a00fdd5cca9251a7fd3c637e6c5ab49c4a8480d7 100644 (file)
@@ -269,4 +269,17 @@ void mail_search_args_result_deserialize(struct mail_search_args *args,
                                         const unsigned char *data,
                                         size_t size);
 
+/* Iterate search_args in batches of messages. The search_args itself is
+   modified each time imap_search_seqset_iter_next() is called. Note that
+   search_args is expected to come from imap_search_get_anyset(), so it should
+   have a single parameter containing SEARCH_ALL, SEARCH_SEQSET or
+   SEARCH_UIDSET. */
+struct mail_search_seqset_iter *
+mail_search_seqset_iter_init(struct mail_search_args *search_args,
+                            uint32_t messages_count, unsigned int batch_size);
+/* Iterate the next batch. Returns TRUE if the batch was updated, FALSE if
+   all the batches have been iterated. */
+bool mail_search_seqset_iter_next(struct mail_search_seqset_iter *iter);
+void mail_search_seqset_iter_deinit(struct mail_search_seqset_iter **iter);
+
 #endif