#include #include #include #include "error_util.h" #include "command-line-arguments.h" /* Search the array of keywords for a given argument, assigning the output variable to the corresponding value. Return FALSE if nothing matches. */ static notmuch_bool_t _process_keyword_arg (const notmuch_opt_desc_t *arg_desc, char next, const char *arg_str) { const notmuch_keyword_t *keywords = arg_desc->keywords; if (next == 0) { /* No keyword given */ arg_str = ""; } while (keywords->name) { if (strcmp (arg_str, keywords->name) == 0) { if (arg_desc->output_var) { *((int *)arg_desc->output_var) = keywords->value; } return TRUE; } keywords++; } if (next != 0) fprintf (stderr, "unknown keyword: %s\n", arg_str); else fprintf (stderr, "option %s needs a keyword\n", arg_desc->name); return FALSE; } static notmuch_bool_t _process_keyword_list (const notmuch_opt_desc_t *arg_desc, const char *arg_str) { char *key_str, *final; unsigned int matched_keys = 0; notmuch_bool_t matched; const notmuch_keyword_t *keywords; key_str = strdup (arg_str); do { keywords = arg_desc->keywords; matched = FALSE; final = strrchr (key_str, ','); if (final) { *final = '\0'; final++; } else { final = key_str; } while (keywords->name && !matched) { if (strcmp (final, keywords->name) == 0) { matched_keys |= keywords->value; matched = TRUE; } keywords++; } if (!matched) { fprintf (stderr, "unknown keyword: \'%s\' in list %s\n", final, arg_str); goto DONE; } } while (final != key_str); if (arg_desc->output_var) { *((unsigned int *)arg_desc->output_var) = matched_keys; } DONE: free (key_str); return matched; } static notmuch_bool_t _process_boolean_arg (const notmuch_opt_desc_t *arg_desc, char next, const char *arg_str) { if (next == 0) { *((notmuch_bool_t *)arg_desc->output_var) = TRUE; return TRUE; } if (strcmp (arg_str, "false") == 0) { *((notmuch_bool_t *)arg_desc->output_var) = FALSE; return TRUE; } if (strcmp (arg_str, "true") == 0) { *((notmuch_bool_t *)arg_desc->output_var) = TRUE; return TRUE; } return FALSE; } /* Search for the {pos_arg_index}th position argument, return FALSE if that does not exist. */ notmuch_bool_t parse_position_arg (const char *arg_str, int pos_arg_index, const notmuch_opt_desc_t *arg_desc) { int pos_arg_counter = 0; while (arg_desc->opt_type != NOTMUCH_OPT_END){ if (arg_desc->opt_type == NOTMUCH_OPT_POSITION) { if (pos_arg_counter == pos_arg_index) { if (arg_desc->output_var) { *((const char **)arg_desc->output_var) = arg_str; } return TRUE; } pos_arg_counter++; } arg_desc++; } return FALSE; } /* * Search for a non-positional (i.e. starting with --) argument matching arg, * parse a possible value, and assign to *output_var */ notmuch_bool_t parse_option (const char *arg, const notmuch_opt_desc_t *options) { assert(arg); assert(options); arg += 2; const notmuch_opt_desc_t *try = options; while (try->opt_type != NOTMUCH_OPT_END) { if (try->name && strncmp (arg, try->name, strlen (try->name)) == 0) { char next = arg[strlen (try->name)]; const char *value= arg+strlen(try->name)+1; char *endptr; /* Everything but boolean arguments (switches) needs a * delimiter, and a non-zero length value. Boolean * arguments may take an optional =true or =false value. */ if (next != '=' && next != ':' && next != 0) return FALSE; if (next == 0) { if (try->opt_type != NOTMUCH_OPT_BOOLEAN && try->opt_type != NOTMUCH_OPT_KEYWORD) return FALSE; } else { if (value[0] == 0) return FALSE; } if (try->output_var == NULL) INTERNAL_ERROR ("output pointer NULL for option %s", try->name); switch (try->opt_type) { case NOTMUCH_OPT_KEYWORD: return _process_keyword_arg (try, next, value); break; case NOTMUCH_OPT_KEYWORD_LIST: return _process_keyword_list (try, value); break; case NOTMUCH_OPT_BOOLEAN: return _process_boolean_arg (try, next, value); break; case NOTMUCH_OPT_INT: *((int *)try->output_var) = strtol (value, &endptr, 10); return (*endptr == 0); break; case NOTMUCH_OPT_STRING: *((const char **)try->output_var) = value; return TRUE; break; case NOTMUCH_OPT_POSITION: case NOTMUCH_OPT_END: default: INTERNAL_ERROR ("unknown or unhandled option type %d", try->opt_type); /*UNREACHED*/ } } try++; } fprintf (stderr, "Unrecognized option: --%s\n", arg); return FALSE; } /* See command-line-arguments.h for description */ int parse_arguments (int argc, char **argv, const notmuch_opt_desc_t *options, int opt_index) { int pos_arg_index = 0; notmuch_bool_t more_args = TRUE; while (more_args && opt_index < argc) { if (strncmp (argv[opt_index],"--",2) != 0) { more_args = parse_position_arg (argv[opt_index], pos_arg_index, options); if (more_args) { pos_arg_index++; opt_index++; } } else { if (strlen (argv[opt_index]) == 2) return opt_index+1; more_args = parse_option (argv[opt_index], options); if (more_args) { opt_index++; } else { opt_index = -1; } } } return opt_index; }