diff --git a/src/lib-sieve/cmd-discard.c b/src/lib-sieve/cmd-discard.c index 92c710951a9f85019bfbb9f357a0a440e7d2e049..a5ed06677527fa4ceff216884b814bf8a2f6a334 100644 --- a/src/lib-sieve/cmd-discard.c +++ b/src/lib-sieve/cmd-discard.c @@ -154,6 +154,7 @@ static int act_discard_commit const struct sieve_action_exec_env *aenv, void *tr_context ATTR_UNUSED, bool *keep) { + aenv->exec_status->significant_action_executed = TRUE; sieve_result_global_log(aenv, "marked message to be discarded if not explicitly delivered " "(discard action)"); diff --git a/src/lib-sieve/cmd-redirect.c b/src/lib-sieve/cmd-redirect.c index 0647ce11d123e523bf2035b53dc5d2783d75d38e..af405b4124fe7232782789370d14f08483257646 100644 --- a/src/lib-sieve/cmd-redirect.c +++ b/src/lib-sieve/cmd-redirect.c @@ -576,6 +576,7 @@ act_redirect_commit(const struct sieve_action *action, sieve_action_duplicate_mark(senv, dupeid, strlen(dupeid), ioloop_time + svinst->redirect_duplicate_period); + aenv->exec_status->significant_action_executed = TRUE; sieve_result_global_log( aenv, "redirect action: forwarded to <%s>", smtp_address_encode(ctx->to_address)); diff --git a/src/lib-sieve/ext-reject.c b/src/lib-sieve/ext-reject.c index 9519f4118dae397ff4d442695e0adeed02dcfdae..def2b8604a3f3b56e60e14cea3a54cafe35a68fc 100644 --- a/src/lib-sieve/ext-reject.c +++ b/src/lib-sieve/ext-reject.c @@ -519,6 +519,7 @@ static int act_reject_commit (aenv, recipient, rj_ctx->reason)) <= 0 ) return ret; + aenv->exec_status->significant_action_executed = TRUE; sieve_result_global_log(aenv, "rejected message from <%s> (%s)", smtp_address_encode(sender), diff --git a/src/lib-sieve/plugins/duplicate/ext-duplicate-common.c b/src/lib-sieve/plugins/duplicate/ext-duplicate-common.c index 62e0bbd291cc462524fa6f7bac78dbc803d186c7..73afea04459f704e60656c10acc95fbabf82e4a4 100644 --- a/src/lib-sieve/plugins/duplicate/ext-duplicate-common.c +++ b/src/lib-sieve/plugins/duplicate/ext-duplicate-common.c @@ -118,6 +118,7 @@ static void act_duplicate_mark_finish /* Message was handled successfully, so track duplicate for this * message. */ + aenv->exec_status->significant_action_executed = TRUE; sieve_action_duplicate_mark (senv, data->hash, sizeof(data->hash), ioloop_time + data->period); } diff --git a/src/lib-sieve/plugins/enotify/cmd-notify.c b/src/lib-sieve/plugins/enotify/cmd-notify.c index e69cbc240a335871e8153d4f2f711546af69b42d..52ef15e2085bc0f6829978d472d3f5e604f3e68e 100644 --- a/src/lib-sieve/plugins/enotify/cmd-notify.c +++ b/src/lib-sieve/plugins/enotify/cmd-notify.c @@ -578,6 +578,8 @@ static int act_notify_commit (aenv->ehandler, NULL, "notify action"); ret = method->def->action_execute(&nenv, act); + if (ret >= 0) + aenv->exec_status->significant_action_executed = TRUE; sieve_error_handler_unref(&nenv.ehandler); } diff --git a/src/lib-sieve/plugins/notify/cmd-notify.c b/src/lib-sieve/plugins/notify/cmd-notify.c index 3019eb9993a78afdcd0264abdcafc7efd2e7cec4..9ee5322ca2761c19fb7a99e4f66c3c7561c76cb3 100644 --- a/src/lib-sieve/plugins/notify/cmd-notify.c +++ b/src/lib-sieve/plugins/notify/cmd-notify.c @@ -838,7 +838,10 @@ static int act_notify_commit result = act_notify_send(aenv, act); } T_END; - return ( result ? SIEVE_EXEC_OK : SIEVE_EXEC_FAILURE ); + if (!result) + return SIEVE_EXEC_FAILURE; + aenv->exec_status->significant_action_executed = TRUE; + return SIEVE_EXEC_OK; } diff --git a/src/lib-sieve/plugins/vacation/cmd-vacation.c b/src/lib-sieve/plugins/vacation/cmd-vacation.c index 2f6806f278951c3287a2cda47cfff8667efcbd86..d7af93e2e76592a61a8660fde71280d5a141ebed 100644 --- a/src/lib-sieve/plugins/vacation/cmd-vacation.c +++ b/src/lib-sieve/plugins/vacation/cmd-vacation.c @@ -1134,6 +1134,7 @@ static int act_vacation_send return SIEVE_EXEC_FAILURE; } + aenv->exec_status->significant_action_executed = TRUE; return SIEVE_EXEC_OK; } @@ -1459,6 +1460,7 @@ static int act_vacation_commit if ( ret == SIEVE_EXEC_OK ) { sieve_number_t seconds; + aenv->exec_status->significant_action_executed = TRUE; sieve_result_global_log(aenv, "sent vacation response to <%s>", smtp_address_encode(sender)); diff --git a/src/lib-sieve/plugins/vnd.dovecot/report/cmd-report.c b/src/lib-sieve/plugins/vnd.dovecot/report/cmd-report.c index 4ec2514f44e731082a1754f4b712ce69333e5553..15caf1ff35be6ba1642835ac95e5698c502f9524 100644 --- a/src/lib-sieve/plugins/vnd.dovecot/report/cmd-report.c +++ b/src/lib-sieve/plugins/vnd.dovecot/report/cmd-report.c @@ -650,6 +650,7 @@ static int act_report_send str_sanitize(error, 512)); } } else { + aenv->exec_status->significant_action_executed = TRUE; sieve_result_global_log(aenv, "sent `%s' report to <%s>", str_sanitize(act->feedback_type, 32), diff --git a/src/lib-sieve/sieve-actions.c b/src/lib-sieve/sieve-actions.c index b71b483712fff1e0b4959e2d1147353f107a41c5..c32b7fd410fbaf761ef4e4024b50e0480838c2bd 100644 --- a/src/lib-sieve/sieve-actions.c +++ b/src/lib-sieve/sieve-actions.c @@ -482,6 +482,32 @@ static struct mail_keywords *act_store_keywords_create return box_keywords; } +static bool +have_equal_keywords(struct mail *mail, struct mail_keywords *new_kw) +{ + const ARRAY_TYPE(keyword_indexes) *old_kw_arr = + mail_get_keyword_indexes(mail); + const unsigned int *old_kw; + unsigned int i, j; + + if (array_count(old_kw_arr) != new_kw->count) + return FALSE; + if (new_kw->count == 0) + return TRUE; + + old_kw = array_front(old_kw_arr); + for (i = 0; i < new_kw->count; i++) { + /* new_kw->count equals old_kw's count and it's easier to use */ + for (j = 0; j < new_kw->count; j++) { + if (old_kw[j] == new_kw->idx[i]) + break; + } + if (j == new_kw->count) + return FALSE; + } + return TRUE; +} + static int act_store_execute (const struct sieve_action *action, const struct sieve_action_exec_env *aenv, void *tr_context) @@ -538,11 +564,17 @@ static int act_store_execute (aenv, &trans->keywords, mail->box, TRUE); if ( keywords != NULL ) { - mail_update_keywords(mail, MODIFY_REPLACE, keywords); + if (!have_equal_keywords(mail, keywords)) { + aenv->exec_status->significant_action_executed = TRUE; + mail_update_keywords(mail, MODIFY_REPLACE, keywords); + } mailbox_keywords_unref(&keywords); } - mail_update_flags(mail, MODIFY_REPLACE, trans->flags); + if ((mail_get_flags(mail) & MAIL_FLAGS_NONRECENT) != trans->flags) { + aenv->exec_status->significant_action_executed = TRUE; + mail_update_flags(mail, MODIFY_REPLACE, trans->flags); + } } return SIEVE_EXEC_OK; @@ -581,7 +613,10 @@ static int act_store_execute if ( trans->flags_altered ) { keywords = act_store_keywords_create(aenv, &trans->keywords, trans->box, FALSE); - mailbox_save_set_flags(save_ctx, trans->flags, keywords); + if (trans->flags != 0 || keywords != NULL) { + aenv->exec_status->significant_action_executed = TRUE; + mailbox_save_set_flags(save_ctx, trans->flags, keywords); + } } else { mailbox_save_copy_flags(save_ctx, mail); } @@ -590,6 +625,8 @@ static int act_store_execute sieve_act_store_get_storage_error(aenv, trans); status = ( trans->error_code == MAIL_ERROR_TEMP ? SIEVE_EXEC_TEMP_FAILURE : SIEVE_EXEC_FAILURE ); + } else { + aenv->exec_status->significant_action_executed = TRUE; } /* Deallocate keywords */ diff --git a/src/lib-sieve/sieve-types.h b/src/lib-sieve/sieve-types.h index 6b93dd203b51ad48cc08888c4418e544d2b17721..dce10d5429f30c3de4799ef3ef8c0d32a63cc11f 100644 --- a/src/lib-sieve/sieve-types.h +++ b/src/lib-sieve/sieve-types.h @@ -260,6 +260,7 @@ struct sieve_exec_status { bool tried_default_save:1; bool keep_original:1; bool store_failed:1; + bool significant_action_executed:1; }; /*