From fd8b7199b71ef84bd453ec875adf462545d24f75 Mon Sep 17 00:00:00 2001 From: Stephan Bosch <stephan@rename-it.nl> Date: Mon, 19 Nov 2007 21:21:27 +0100 Subject: [PATCH] Revised positional argument checking and fixed the validator's error handling. --- sieve/errors/address-errors.sieve | 2 + sieve/errors/header-errors.sieve | 2 + src/lib-sieve/cmd-redirect.c | 11 +-- src/lib-sieve/cmd-require.c | 6 +- src/lib-sieve/ext-envelope.c | 17 ++--- src/lib-sieve/ext-fileinto.c | 11 ++- src/lib-sieve/ext-reject.c | 4 + .../plugins/subaddress/subaddress.sieve | 2 +- src/lib-sieve/plugins/vacation/ext-vacation.c | 7 +- src/lib-sieve/sieve-ast.c | 4 +- src/lib-sieve/sieve-ast.h | 4 +- src/lib-sieve/sieve-common.h | 2 + src/lib-sieve/sieve-validator.c | 74 ++++++++++++------- src/lib-sieve/sieve-validator.h | 5 ++ src/lib-sieve/tst-address.c | 73 +++++++++--------- src/lib-sieve/tst-exists.c | 8 +- src/lib-sieve/tst-header.c | 55 +++++++------- src/lib-sieve/tst-size.c | 8 +- 18 files changed, 164 insertions(+), 131 deletions(-) diff --git a/sieve/errors/address-errors.sieve b/sieve/errors/address-errors.sieve index 603320c83..4744f1d9c 100644 --- a/sieve/errors/address-errors.sieve +++ b/sieve/errors/address-errors.sieve @@ -1,3 +1,5 @@ +require "comparator-i;ascii-numeric"; + if address :isnot :comparator "i;ascii-casemap" :localpart "From" "nico" { discard; } diff --git a/sieve/errors/header-errors.sieve b/sieve/errors/header-errors.sieve index 6c66b5ac3..ac6108ba1 100644 --- a/sieve/errors/header-errors.sieve +++ b/sieve/errors/header-errors.sieve @@ -1,3 +1,5 @@ +require "comparator-i;ascii-numeric"; + if header :isnot :comparator "i;ascii-casemap" "From" "nico" { discard; } diff --git a/src/lib-sieve/cmd-redirect.c b/src/lib-sieve/cmd-redirect.c index 7143599bf..a45076a2b 100644 --- a/src/lib-sieve/cmd-redirect.c +++ b/src/lib-sieve/cmd-redirect.c @@ -6,23 +6,20 @@ bool cmd_redirect_validate(struct sieve_validator *validator, struct sieve_command_context *cmd) { - struct sieve_ast_argument *argument; + struct sieve_ast_argument *arg; /* Check valid syntax * Syntax: redirect <address: string> */ - if ( !sieve_validate_command_arguments(validator, cmd, 1, &argument) || + if ( !sieve_validate_command_arguments(validator, cmd, 1, &arg) || !sieve_validate_command_subtests(validator, cmd, 0) || !sieve_validate_command_block(validator, cmd, FALSE, FALSE) ) { return FALSE; } /* Check argument */ - if ( sieve_ast_argument_type(argument) != SAAT_STRING ) { - /* Somethin else */ - sieve_command_validate_error(validator, cmd, - "the redirect command accepts a single string argument (address) but %s was found", - sieve_ast_argument_name(argument)); + if ( !sieve_validate_positional_argument + (validator, cmd, arg, "address", 1, SAAT_STRING) ) { return FALSE; } diff --git a/src/lib-sieve/cmd-require.c b/src/lib-sieve/cmd-require.c index 738f6f826..ed03974bd 100644 --- a/src/lib-sieve/cmd-require.c +++ b/src/lib-sieve/cmd-require.c @@ -21,7 +21,8 @@ bool cmd_require_validate(struct sieve_validator *validator, struct sieve_comman !sieve_ast_prev_cmd_is(cmd->ast_node, "require") ) ) { sieve_command_validate_error(validator, cmd, - "the require command can only be placed at top level at the beginning of the file"); + "the require command can only be placed at top level " + "at the beginning of the file"); return FALSE; } @@ -61,7 +62,8 @@ bool cmd_require_validate(struct sieve_validator *validator, struct sieve_comman } else { /* Something else */ sieve_command_validate_error(validator, cmd, - "the require command accepts a single string or string list argument but %s was found", + "the require command accepts a single string or string list argument, " + "but %s was found", sieve_ast_argument_name(arg)); return FALSE; } diff --git a/src/lib-sieve/ext-envelope.c b/src/lib-sieve/ext-envelope.c index 12bcae1ba..d3447983a 100644 --- a/src/lib-sieve/ext-envelope.c +++ b/src/lib-sieve/ext-envelope.c @@ -88,20 +88,17 @@ static bool tst_envelope_validate(struct sieve_validator *validator, struct siev return FALSE; } - if ( sieve_ast_argument_type(arg) != SAAT_STRING && sieve_ast_argument_type(arg) != SAAT_STRING_LIST ) { - sieve_command_validate_error(validator, tst, - "the envelope test expects a string-list as first argument (envelope part), but %s was found", - sieve_ast_argument_name(arg)); - return FALSE; + if ( !sieve_validate_positional_argument + (validator, tst, arg, "envelope part", 1, SAAT_STRING_LIST) ) { + return FALSE; } sieve_validator_argument_activate(validator, arg); arg = sieve_ast_argument_next(arg); - if ( sieve_ast_argument_type(arg) != SAAT_STRING && sieve_ast_argument_type(arg) != SAAT_STRING_LIST ) { - sieve_command_validate_error(validator, tst, - "the envelope test expects a string-list as second argument (key list), but %s was found", - sieve_ast_argument_name(arg)); - return FALSE; + + if ( !sieve_validate_positional_argument + (validator, tst, arg, "key list", 2, SAAT_STRING_LIST) ) { + return FALSE; } sieve_validator_argument_activate(validator, arg); diff --git a/src/lib-sieve/ext-fileinto.c b/src/lib-sieve/ext-fileinto.c index 42d0cbb53..d811b56d0 100644 --- a/src/lib-sieve/ext-fileinto.c +++ b/src/lib-sieve/ext-fileinto.c @@ -51,7 +51,7 @@ static bool cmd_fileinto_validate(struct sieve_validator *validator, struct siev struct sieve_ast_argument *arg; /* Check valid syntax: - * reject <reason: string> + * fileinto <folder: string> */ if ( !sieve_validate_command_arguments(validator, cmd, 1, &arg) || !sieve_validate_command_subtests(validator, cmd, 0) || @@ -60,7 +60,10 @@ static bool cmd_fileinto_validate(struct sieve_validator *validator, struct siev return FALSE; } - + if ( !sieve_validate_positional_argument + (validator, cmd, arg, "folder", 1, SAAT_STRING) ) { + return FALSE; + } sieve_validator_argument_activate(validator, arg); return TRUE; @@ -85,8 +88,8 @@ static bool cmd_fileinto_generate sieve_generator_emit_opcode_ext(generator, ext_my_id); /* Generate arguments */ - if ( !sieve_generate_arguments(generator, ctx, NULL) ) - return FALSE; + if ( !sieve_generate_arguments(generator, ctx, NULL) ) + return FALSE; return TRUE; } diff --git a/src/lib-sieve/ext-reject.c b/src/lib-sieve/ext-reject.c index 141123a02..bedc71c0e 100644 --- a/src/lib-sieve/ext-reject.c +++ b/src/lib-sieve/ext-reject.c @@ -68,6 +68,10 @@ static bool cmd_reject_validate(struct sieve_validator *validator, struct sieve_ return FALSE; } + if ( !sieve_validate_positional_argument + (validator, cmd, arg, "reason", 1, SAAT_STRING) ) { + return FALSE; + } sieve_validator_argument_activate(validator, arg); return TRUE; diff --git a/src/lib-sieve/plugins/subaddress/subaddress.sieve b/src/lib-sieve/plugins/subaddress/subaddress.sieve index 1b4a0c715..7c2ab9329 100644 --- a/src/lib-sieve/plugins/subaddress/subaddress.sieve +++ b/src/lib-sieve/plugins/subaddress/subaddress.sieve @@ -8,7 +8,7 @@ if address :comparator "i;ascii-casemap" :user "from" "STEPHAN" { keep; stop; } - fileinto "INBOX.frop"; + stop; } keep; diff --git a/src/lib-sieve/plugins/vacation/ext-vacation.c b/src/lib-sieve/plugins/vacation/ext-vacation.c index 085a1c5d1..d80c6392a 100644 --- a/src/lib-sieve/plugins/vacation/ext-vacation.c +++ b/src/lib-sieve/plugins/vacation/ext-vacation.c @@ -244,7 +244,8 @@ static bool cmd_vacation_registered(struct sieve_validator *validator, struct si /* Command validation */ -static bool cmd_vacation_validate(struct sieve_validator *validator, struct sieve_command_context *cmd) +static bool cmd_vacation_validate(struct sieve_validator *validator, + struct sieve_command_context *cmd) { struct sieve_ast_argument *arg; @@ -260,6 +261,10 @@ static bool cmd_vacation_validate(struct sieve_validator *validator, struct siev return FALSE; } + if ( !sieve_validate_positional_argument + (validator, cmd, arg, "reason", 1, SAAT_STRING) ) { + return FALSE; + } sieve_validator_argument_activate(validator, arg); return TRUE; diff --git a/src/lib-sieve/sieve-ast.c b/src/lib-sieve/sieve-ast.c index b66658ac7..255d29bbf 100644 --- a/src/lib-sieve/sieve-ast.c +++ b/src/lib-sieve/sieve-ast.c @@ -243,8 +243,8 @@ struct sieve_ast_argument *sieve_ast_arguments_delete return sieve_ast_arg_list_delete(first, count); } -const char *sieve_ast_argument_name(struct sieve_ast_argument *argument) { - switch ( argument->type ) { +const char *sieve_ast_argument_type_name(enum sieve_ast_argument_type arg_type) { + switch ( arg_type ) { case SAAT_NONE: return "none"; case SAAT_STRING_LIST: return "a string list"; diff --git a/src/lib-sieve/sieve-ast.h b/src/lib-sieve/sieve-ast.h index bfc60ec3e..6c7a65727 100644 --- a/src/lib-sieve/sieve-ast.h +++ b/src/lib-sieve/sieve-ast.h @@ -163,7 +163,9 @@ struct sieve_ast_argument *sieve_ast_argument_number_create struct sieve_ast_argument *sieve_ast_arguments_delete (struct sieve_ast_argument *first, unsigned int count); -const char *sieve_ast_argument_name(struct sieve_ast_argument *argument); +const char *sieve_ast_argument_type_name(enum sieve_ast_argument_type arg_type); +#define sieve_ast_argument_name(argument) \ + sieve_ast_argument_type_name((argument)->type) void sieve_ast_stringlist_add (struct sieve_ast_argument *list, const string_t *str, unsigned int source_line); diff --git a/src/lib-sieve/sieve-common.h b/src/lib-sieve/sieve-common.h index e60ee85d3..cb4994c62 100644 --- a/src/lib-sieve/sieve-common.h +++ b/src/lib-sieve/sieve-common.h @@ -15,6 +15,8 @@ typedef uint64_t sieve_number_t; */ /* sieve-ast.h */ +enum sieve_ast_argument_type; + struct sieve_ast; struct sieve_ast_node; struct sieve_ast_argument; diff --git a/src/lib-sieve/sieve-validator.c b/src/lib-sieve/sieve-validator.c index 2c2cd3f39..aeb39f2a5 100644 --- a/src/lib-sieve/sieve-validator.c +++ b/src/lib-sieve/sieve-validator.c @@ -329,7 +329,48 @@ inline const void *sieve_validator_extension_get_context(struct sieve_validator return array_idx(&validator->ext_contexts, (unsigned int) ext_id); } -/* Tag Validation API */ +/* Argument Validation API */ + +bool sieve_validate_positional_argument + (struct sieve_validator *validator, struct sieve_command_context *cmd, + struct sieve_ast_argument *arg, const char *arg_name, unsigned int arg_pos, + enum sieve_ast_argument_type req_type) +{ + if ( sieve_ast_argument_type(arg) != req_type && + (sieve_ast_argument_type(arg) != SAAT_STRING || + req_type != SAAT_STRING_LIST) ) + { + sieve_command_validate_error(validator, cmd, + "the %s %s expects %s as argument %d (%s), but %s was found", + cmd->command->identifier, sieve_command_type_name(cmd->command), + sieve_ast_argument_type_name(req_type), + arg_pos, arg_name, sieve_ast_argument_name(arg)); + return FALSE; + } + + return TRUE; +} + +void sieve_validator_argument_activate + (struct sieve_validator *validator ATTR_UNUSED, struct sieve_ast_argument *arg) +{ + switch ( sieve_ast_argument_type(arg) ) { + case SAAT_NUMBER: + arg->argument = &number_argument; + break; + case SAAT_STRING: + arg->argument = &string_argument; + break; + case SAAT_STRING_LIST: + arg->argument = &string_list_argument; + break; + case SAAT_TAG: + i_error("!!BUG!!: sieve_validator_argument_activate: cannot activate tagged argument."); + break; + default: + break; + } +} /* Test validation API */ @@ -428,27 +469,6 @@ bool sieve_validate_command_arguments return TRUE; } - -void sieve_validator_argument_activate - (struct sieve_validator *validator ATTR_UNUSED, struct sieve_ast_argument *arg) -{ - switch ( sieve_ast_argument_type(arg) ) { - case SAAT_NUMBER: - arg->argument = &number_argument; - break; - case SAAT_STRING: - arg->argument = &string_argument; - break; - case SAAT_STRING_LIST: - arg->argument = &string_list_argument; - break; - case SAAT_TAG: - i_error("!!BUG!!: sieve_validator_argument_activate: cannot activate tagged argument."); - break; - default: - break; - } -} /* Command Validation API */ @@ -580,15 +600,16 @@ static bool sieve_validate_test(struct sieve_validator *validator, struct sieve_ static bool sieve_validate_test_list(struct sieve_validator *validator, struct sieve_ast_node *test_list) { + bool result = TRUE; struct sieve_ast_node *test; test = sieve_ast_test_first(test_list); while ( test != NULL ) { - sieve_validate_test(validator, test); + result = result && sieve_validate_test(validator, test); test = sieve_ast_test_next(test); } - return TRUE; + return result; } static bool sieve_validate_command(struct sieve_validator *validator, struct sieve_ast_node *cmd_node) @@ -646,17 +667,18 @@ static bool sieve_validate_command(struct sieve_validator *validator, struct sie static bool sieve_validate_block(struct sieve_validator *validator, struct sieve_ast_node *block) { + bool result = TRUE; struct sieve_ast_node *command; t_push(); command = sieve_ast_command_first(block); while ( command != NULL ) { - sieve_validate_command(validator, command); + result = result && sieve_validate_command(validator, command); command = sieve_ast_command_next(command); } t_pop(); - return TRUE; + return result; } bool sieve_validator_run(struct sieve_validator *validator) { diff --git a/src/lib-sieve/sieve-validator.h b/src/lib-sieve/sieve-validator.h index f980b6dea..601e5200f 100644 --- a/src/lib-sieve/sieve-validator.h +++ b/src/lib-sieve/sieve-validator.h @@ -37,9 +37,14 @@ void sieve_validator_link_match_type_tags unsigned int id_code); /* Argument validation */ + bool sieve_validate_command_arguments (struct sieve_validator *validator, struct sieve_command_context *tst, const unsigned int count, struct sieve_ast_argument **first_positional); +bool sieve_validate_positional_argument + (struct sieve_validator *validator, struct sieve_command_context *cmd, + struct sieve_ast_argument *arg, const char *arg_name, unsigned int arg_pos, + enum sieve_ast_argument_type req_type); void sieve_validator_argument_activate (struct sieve_validator *validator, struct sieve_ast_argument *arg); diff --git a/src/lib-sieve/tst-address.c b/src/lib-sieve/tst-address.c index cb65d0c71..69ef9c038 100644 --- a/src/lib-sieve/tst-address.c +++ b/src/lib-sieve/tst-address.c @@ -58,20 +58,17 @@ bool tst_address_validate(struct sieve_validator *validator, struct sieve_comman tst->data = arg; - if ( sieve_ast_argument_type(arg) != SAAT_STRING && sieve_ast_argument_type(arg) != SAAT_STRING_LIST ) { - sieve_command_validate_error(validator, tst, - "the address test expects a string-list as first argument (header list), but %s was found", - sieve_ast_argument_name(arg)); - return FALSE; + if ( !sieve_validate_positional_argument + (validator, tst, arg, "header list", 1, SAAT_STRING_LIST) ) { + return FALSE; } sieve_validator_argument_activate(validator, arg); - + arg = sieve_ast_argument_next(arg); - if ( sieve_ast_argument_type(arg) != SAAT_STRING && sieve_ast_argument_type(arg) != SAAT_STRING_LIST ) { - sieve_command_validate_error(validator, tst, - "the address test expects a string-list as second argument (key list), but %s was found", - sieve_ast_argument_name(arg)); - return FALSE; + + if ( !sieve_validate_positional_argument + (validator, tst, arg, "key list", 2, SAAT_STRING_LIST) ) { + return FALSE; } sieve_validator_argument_activate(validator, arg); @@ -104,26 +101,26 @@ static bool tst_address_opcode_dump printf("ADDRESS\n"); /* Handle any optional arguments */ - if ( sieve_operand_optional_present(sbin, address) ) { - while ( (opt_code=sieve_operand_optional_read(sbin, address)) ) { - switch ( opt_code ) { - case OPT_COMPARATOR: + if ( sieve_operand_optional_present(sbin, address) ) { + while ( (opt_code=sieve_operand_optional_read(sbin, address)) ) { + switch ( opt_code ) { + case OPT_COMPARATOR: if ( !sieve_opr_comparator_dump(interp, sbin, address) ) return FALSE; - break; - case OPT_MATCH_TYPE: + break; + case OPT_MATCH_TYPE: if ( !sieve_opr_match_type_dump(interp, sbin, address) ) - return FALSE; - break; + return FALSE; + break; case OPT_ADDRESS_PART: if ( !sieve_opr_address_part_dump(interp, sbin, address) ) return FALSE; break; - default: - return FALSE; - } - } - } + default: + return FALSE; + } + } + } return sieve_opr_stringlist_dump(sbin, address) && @@ -149,26 +146,26 @@ static bool tst_address_opcode_execute printf("?? ADDRESS\n"); /* Handle any optional arguments */ - if ( sieve_operand_optional_present(sbin, address) ) { - while ( (opt_code=sieve_operand_optional_read(sbin, address)) ) { - switch ( opt_code ) { - case OPT_COMPARATOR: - if ( (cmp = sieve_opr_comparator_read(interp, sbin, address)) == NULL ) + if ( sieve_operand_optional_present(sbin, address) ) { + while ( (opt_code=sieve_operand_optional_read(sbin, address)) ) { + switch ( opt_code ) { + case OPT_COMPARATOR: + if ( (cmp = sieve_opr_comparator_read(interp, sbin, address)) == NULL ) return FALSE; - break; - case OPT_MATCH_TYPE: - if ( (mtch = sieve_opr_match_type_read(interp, sbin, address)) == NULL ) + break; + case OPT_MATCH_TYPE: + if ( (mtch = sieve_opr_match_type_read(interp, sbin, address)) == NULL ) return FALSE; - break; + break; case OPT_ADDRESS_PART: if ( (addrp = sieve_opr_address_part_read(interp, sbin, address)) == NULL ) return FALSE; break; - default: - return FALSE; - } - } - } + default: + return FALSE; + } + } + } t_push(); diff --git a/src/lib-sieve/tst-exists.c b/src/lib-sieve/tst-exists.c index e515c089e..d5c1d3796 100644 --- a/src/lib-sieve/tst-exists.c +++ b/src/lib-sieve/tst-exists.c @@ -30,11 +30,9 @@ bool tst_exists_validate(struct sieve_validator *validator, struct sieve_command return FALSE; } - if ( sieve_ast_argument_type(arg) != SAAT_STRING && sieve_ast_argument_type(arg) != SAAT_STRING_LIST ) { - sieve_command_validate_error(validator, tst, - "the exists test expects a string-list as only argument (header names), but %s was found", - sieve_ast_argument_name(arg)); - return FALSE; + if ( !sieve_validate_positional_argument + (validator, tst, arg, "header names", 1, SAAT_STRING_LIST) ) { + return FALSE; } sieve_validator_argument_activate(validator, arg); diff --git a/src/lib-sieve/tst-header.c b/src/lib-sieve/tst-header.c index ff42d2484..30567c032 100644 --- a/src/lib-sieve/tst-header.c +++ b/src/lib-sieve/tst-header.c @@ -51,20 +51,17 @@ bool tst_header_validate(struct sieve_validator *validator, struct sieve_command !sieve_validate_command_subtests(validator, tst, 0) ) return FALSE; - if ( sieve_ast_argument_type(arg) != SAAT_STRING && sieve_ast_argument_type(arg) != SAAT_STRING_LIST ) { - sieve_command_validate_error(validator, tst, - "the header test expects a string-list as first argument (header names), but %s was found", - sieve_ast_argument_name(arg)); - return FALSE; + if ( !sieve_validate_positional_argument + (validator, tst, arg, "header names", 1, SAAT_STRING_LIST) ) { + return FALSE; } sieve_validator_argument_activate(validator, arg); arg = sieve_ast_argument_next(arg); - if ( sieve_ast_argument_type(arg) != SAAT_STRING && sieve_ast_argument_type(arg) != SAAT_STRING_LIST ) { - sieve_command_validate_error(validator, tst, - "the header test expects a string-list as second argument (key list), but %s was found", - sieve_ast_argument_name(arg)); - return FALSE; + + if ( !sieve_validate_positional_argument + (validator, tst, arg, "key list", 2, SAAT_STRING_LIST) ) { + return FALSE; } sieve_validator_argument_activate(validator, arg); @@ -79,8 +76,8 @@ bool tst_header_generate sieve_generator_emit_opcode(generator, SIEVE_OPCODE_HEADER); /* Generate arguments */ - if ( !sieve_generate_arguments(generator, ctx, NULL) ) - return FALSE; + if ( !sieve_generate_arguments(generator, ctx, NULL) ) + return FALSE; return TRUE; } @@ -93,7 +90,7 @@ static bool tst_header_opcode_dump { unsigned int opt_code; - printf("HEADER\n"); + printf("HEADER\n"); /* Handle any optional arguments */ if ( sieve_operand_optional_present(sbin, address) ) { @@ -112,8 +109,8 @@ static bool tst_header_opcode_dump } return - sieve_opr_stringlist_dump(sbin, address) && - sieve_opr_stringlist_dump(sbin, address); + sieve_opr_stringlist_dump(sbin, address) && + sieve_opr_stringlist_dump(sbin, address); } /* Code execution */ @@ -134,20 +131,20 @@ static bool tst_header_opcode_execute printf("?? HEADER\n"); /* Handle any optional arguments */ - if ( sieve_operand_optional_present(sbin, address) ) { - while ( (opt_code=sieve_operand_optional_read(sbin, address)) ) { - switch ( opt_code ) { - case OPT_COMPARATOR: - cmp = sieve_opr_comparator_read(interp, sbin, address); - break; - case OPT_MATCH_TYPE: - mtch = sieve_opr_match_type_read(interp, sbin, address); - break; - default: - return FALSE; - } - } - } + if ( sieve_operand_optional_present(sbin, address) ) { + while ( (opt_code=sieve_operand_optional_read(sbin, address)) ) { + switch ( opt_code ) { + case OPT_COMPARATOR: + cmp = sieve_opr_comparator_read(interp, sbin, address); + break; + case OPT_MATCH_TYPE: + mtch = sieve_opr_match_type_read(interp, sbin, address); + break; + default: + return FALSE; + } + } + } t_push(); diff --git a/src/lib-sieve/tst-size.c b/src/lib-sieve/tst-size.c index 2ddca289c..5823e4ca1 100644 --- a/src/lib-sieve/tst-size.c +++ b/src/lib-sieve/tst-size.c @@ -114,11 +114,9 @@ bool tst_size_validate(struct sieve_validator *validator, struct sieve_command_c return FALSE; } - if ( sieve_ast_argument_type(arg) != SAAT_NUMBER ) { - sieve_command_validate_error(validator, tst, - "the size test expects a number as argument (limit), but %s was found", - sieve_ast_argument_name(arg)); - return FALSE; + if ( !sieve_validate_positional_argument + (validator, tst, arg, "limit", 1, SAAT_NUMBER) ) { + return FALSE; } sieve_validator_argument_activate(validator, arg); -- GitLab