Skip to content
Snippets Groups Projects
Commit e3b369ee authored by Stephan Bosch's avatar Stephan Bosch
Browse files

Implemented support for side-effects to implicit keep and finished the imapflags extension.

parent 1a10419a
No related branches found
No related tags found
No related merge requests found
...@@ -149,15 +149,15 @@ Extensions and their implementation status: ...@@ -149,15 +149,15 @@ Extensions and their implementation status:
encoded-character: full encoded-character: full
Other RFCs/drafts: Other RFCs/drafts:
subaddress: full, but not configurable subaddress: mostly full; not configurable
comparator-i;ascii-numeric: full comparator-i;ascii-numeric: full
relational: full relational: full
copy: full copy: full
regex: full, but suboptimal and no UTF-8 regex: mostly full; but suboptimal and no UTF-8
body: full, but text body-transform implementation is simple body: full, but text body-transform implementation is simple
include: almost full; needs some more work (no external binaries) include: mostly full; needs some more work (no external binaries)
vacation: almost full; no support for required References header vacation: mostly full; no support for required References header
imapflags: flag management works, but flags are not stored imapflags: full
variables: mostly full; currently no support for future namespaces variables: mostly full; currently no support for future namespaces
Low priority: Low priority:
......
Next (in order of descending priority/precedence): Next (in order of descending priority/precedence):
* Finish implementing all extensions supported by cmusieve, except notify.
* Get rid of all <stdio.h> printf()s in the library; use trace macro instead * Get rid of all <stdio.h> printf()s in the library; use trace macro instead
* Revise extension support for comparators, match-types, address-parts and * Revise extension support for comparators, match-types, address-parts and
side-effects. Current implementation emits redundant bytes and associated code side-effects. Current implementation emits redundant bytes and associated code
...@@ -7,7 +6,6 @@ Next (in order of descending priority/precedence): ...@@ -7,7 +6,6 @@ Next (in order of descending priority/precedence):
* Emit line numbers for certain action commands to provide useful runtime error * Emit line numbers for certain action commands to provide useful runtime error
messages. messages.
* Improve handling of old/corrupt binaries. * Improve handling of old/corrupt binaries.
* Handle persistent side-effects for the (implicit) keep action
* Full security review. Enforce limits on number of created objects, script * Full security review. Enforce limits on number of created objects, script
size, execution time, etc... size, execution time, etc...
......
...@@ -3,8 +3,8 @@ ...@@ -3,8 +3,8 @@
* *
* Authors: Stephan Bosch * Authors: Stephan Bosch
* Specification: draft-ietf-sieve-imapflags-05 * Specification: draft-ietf-sieve-imapflags-05
* Implementation: flag management works, not stored though. * Implementation: full
* Status: under development * Status: experimental, largely untested
* *
*/ */
......
...@@ -25,4 +25,4 @@ if hasflag :count "ge" :comparator "i;ascii-numeric" "2" { ...@@ -25,4 +25,4 @@ if hasflag :count "ge" :comparator "i;ascii-numeric" "2" {
fileinto "imap-twoflags"; fileinto "imap-twoflags";
} }
fileinto :flags "\\Seen MDNRequired \\Draft" "INBOX"; fileinto :flags "MDNRequired \\Draft" "INBOX";
...@@ -235,7 +235,7 @@ static void seff_flags_print ...@@ -235,7 +235,7 @@ static void seff_flags_print
(const struct sieve_side_effect *seffect ATTR_UNUSED, (const struct sieve_side_effect *seffect ATTR_UNUSED,
const struct sieve_action *action ATTR_UNUSED, const struct sieve_action *action ATTR_UNUSED,
struct sieve_result *result, struct sieve_result *result,
void *se_context ATTR_UNUSED, bool *keep) void *se_context ATTR_UNUSED, bool *keep ATTR_UNUSED)
{ {
struct seff_flags_context *ctx = (struct seff_flags_context *) se_context; struct seff_flags_context *ctx = (struct seff_flags_context *) se_context;
unsigned int i; unsigned int i;
...@@ -243,31 +243,31 @@ static void seff_flags_print ...@@ -243,31 +243,31 @@ static void seff_flags_print
if ( ctx == NULL ) if ( ctx == NULL )
ctx = seff_flags_get_implicit_context(result); ctx = seff_flags_get_implicit_context(result);
printf(" + add flags:"); if ( ctx->flags != 0 || array_count(&ctx->keywords) ) {
printf(" + add flags:");
if ( (ctx->flags & MAIL_FLAGGED) > 0 ) if ( (ctx->flags & MAIL_FLAGGED) > 0 )
printf(" \\flagged\n"); printf(" \\flagged\n");
if ( (ctx->flags & MAIL_ANSWERED) > 0 ) if ( (ctx->flags & MAIL_ANSWERED) > 0 )
printf(" \\answered"); printf(" \\answered");
if ( (ctx->flags & MAIL_DELETED) > 0 ) if ( (ctx->flags & MAIL_DELETED) > 0 )
printf(" \\deleted"); printf(" \\deleted");
if ( (ctx->flags & MAIL_SEEN) > 0 ) if ( (ctx->flags & MAIL_SEEN) > 0 )
printf(" \\seen"); printf(" \\seen");
if ( (ctx->flags & MAIL_DRAFT) > 0 ) if ( (ctx->flags & MAIL_DRAFT) > 0 )
printf(" \\draft"); printf(" \\draft");
for ( i = 0; i < array_count(&ctx->keywords); i++ ) { for ( i = 0; i < array_count(&ctx->keywords); i++ ) {
const char *const *keyword = array_idx(&ctx->keywords, i); const char *const *keyword = array_idx(&ctx->keywords, i);
printf(" %s", *keyword); printf(" %s", *keyword);
}; };
}
printf("\n"); printf("\n");
*keep = TRUE;
} }
static bool seff_flags_pre_execute static bool seff_flags_pre_execute
......
...@@ -329,26 +329,71 @@ bool sieve_result_print(struct sieve_result *result) ...@@ -329,26 +329,71 @@ bool sieve_result_print(struct sieve_result *result)
return TRUE; return TRUE;
} }
static bool sieve_result_implicit_keep(struct sieve_result *result) static bool sieve_result_implicit_keep
(struct sieve_result *result, bool rollback)
{ {
bool success = TRUE; bool success = TRUE;
bool dummy = TRUE; bool dummy = TRUE;
struct act_store_context ctx; struct act_store_context ctx;
struct sieve_result_side_effect *rsef, *rsef_first = NULL;
void *tr_context; void *tr_context;
ctx.folder = result->action_env.scriptenv->inbox; ctx.folder = result->action_env.scriptenv->inbox;
/* FIXME: Handle persistent side-effects for the (implicit) keep action */ /* Also apply any implicit side effects if applicable */
if ( !rollback && result->implicit_seffects != NULL ) {
struct sieve_result_implicit_side_effects *implseff;
/* Check for implicit side effects to store action */
implseff = (struct sieve_result_implicit_side_effects *)
hash_lookup(result->implicit_seffects, &act_store);
if ( implseff != NULL && implseff->seffects != NULL )
rsef_first = implseff->seffects->first_effect;
}
success = act_store.start success = act_store.start
(&act_store, &result->action_env, (void *) &ctx, &tr_context); (&act_store, &result->action_env, (void *) &ctx, &tr_context);
rsef = rsef_first;
while ( rsef != NULL ) {
const struct sieve_side_effect *sef = rsef->seffect;
if ( sef->pre_execute != NULL )
success = success & sef->pre_execute
(sef, &act_store, &result->action_env, &rsef->context, tr_context);
rsef = rsef->next;
}
success = success && act_store.execute success = success && act_store.execute
(&act_store, &result->action_env, tr_context); (&act_store, &result->action_env, tr_context);
rsef = rsef_first;
while ( rsef != NULL ) {
const struct sieve_side_effect *sef = rsef->seffect;
if ( sef->post_execute != NULL )
success = success && sef->post_execute
(sef, &act_store, &result->action_env, rsef->context, tr_context);
rsef = rsef->next;
}
if ( success ) { if ( success ) {
return act_store.commit success = act_store.commit
(&act_store, &result->action_env, tr_context, &dummy); (&act_store, &result->action_env, tr_context, &dummy);
rsef = rsef_first;
while ( rsef != NULL ) {
const struct sieve_side_effect *sef = rsef->seffect;
bool keep = TRUE;
if ( sef->post_commit != NULL )
sef->post_commit
(sef, &act_store, &result->action_env, rsef->context, tr_context,
&keep);
rsef = rsef->next;
}
return success;
} }
act_store.rollback(&act_store, &result->action_env, tr_context, success); act_store.rollback(&act_store, &result->action_env, tr_context, success);
...@@ -494,7 +539,7 @@ int sieve_result_execute ...@@ -494,7 +539,7 @@ int sieve_result_execute
if ( !commit_ok || implicit_keep ) { if ( !commit_ok || implicit_keep ) {
printf("Executing implicit keep\n"); printf("Executing implicit keep\n");
if ( !sieve_result_implicit_keep(result) ) if ( !sieve_result_implicit_keep(result, !commit_ok) )
return -1; return -1;
return ( commit_ok ? 1 /* Success */ : 0 /* Implicit keep executed */ ); return ( commit_ok ? 1 /* Success */ : 0 /* Implicit keep executed */ );
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment

Consent

On this website, we use the web analytics service Matomo to analyze and review the use of our website. Through the collected statistics, we can improve our offerings and make them more appealing for you. Here, you can decide whether to allow us to process your data and set corresponding cookies for these purposes, in addition to technically necessary cookies. Further information on data protection—especially regarding "cookies" and "Matomo"—can be found in our privacy policy. You can withdraw your consent at any time.