mirror of
https://git.notmuchmail.org/git/notmuch
synced 2024-11-21 18:38:08 +01:00
lib: config list iterators
Since xapian provides the ability to restrict the iterator to a given prefix, we expose this ability to the user. Otherwise we mimic the other iterator interfances in notmuch (e.g. tags.c).
This commit is contained in:
parent
3458e3c89c
commit
92e59568fa
3 changed files with 210 additions and 0 deletions
106
lib/config.cc
106
lib/config.cc
|
@ -24,6 +24,21 @@
|
||||||
|
|
||||||
static const std::string CONFIG_PREFIX = "C";
|
static const std::string CONFIG_PREFIX = "C";
|
||||||
|
|
||||||
|
struct _notmuch_config_list {
|
||||||
|
notmuch_database_t *notmuch;
|
||||||
|
Xapian::TermIterator iterator;
|
||||||
|
char *current_key;
|
||||||
|
char *current_val;
|
||||||
|
};
|
||||||
|
|
||||||
|
static int
|
||||||
|
_notmuch_config_list_destroy (notmuch_config_list_t *list)
|
||||||
|
{
|
||||||
|
/* invoke destructor w/o deallocating memory */
|
||||||
|
list->iterator.~TermIterator();
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
notmuch_status_t
|
notmuch_status_t
|
||||||
notmuch_database_set_config (notmuch_database_t *notmuch,
|
notmuch_database_set_config (notmuch_database_t *notmuch,
|
||||||
const char *key,
|
const char *key,
|
||||||
|
@ -85,3 +100,94 @@ notmuch_database_get_config (notmuch_database_t *notmuch,
|
||||||
|
|
||||||
return NOTMUCH_STATUS_SUCCESS;
|
return NOTMUCH_STATUS_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
notmuch_status_t
|
||||||
|
notmuch_database_get_config_list (notmuch_database_t *notmuch,
|
||||||
|
const char *prefix,
|
||||||
|
notmuch_config_list_t **out)
|
||||||
|
{
|
||||||
|
notmuch_config_list_t *list = NULL;
|
||||||
|
notmuch_status_t status = NOTMUCH_STATUS_SUCCESS;
|
||||||
|
|
||||||
|
list = talloc (notmuch, notmuch_config_list_t);
|
||||||
|
if (! list) {
|
||||||
|
status = NOTMUCH_STATUS_OUT_OF_MEMORY;
|
||||||
|
goto DONE;
|
||||||
|
}
|
||||||
|
|
||||||
|
talloc_set_destructor (list, _notmuch_config_list_destroy);
|
||||||
|
list->notmuch = notmuch;
|
||||||
|
list->current_key = NULL;
|
||||||
|
list->current_val = NULL;
|
||||||
|
|
||||||
|
try {
|
||||||
|
|
||||||
|
new(&(list->iterator)) Xapian::TermIterator (notmuch->xapian_db->metadata_keys_begin
|
||||||
|
(CONFIG_PREFIX + (prefix ? prefix : "")));
|
||||||
|
|
||||||
|
} catch (const Xapian::Error &error) {
|
||||||
|
_notmuch_database_log (notmuch, "A Xapian exception occurred getting metadata iterator: %s.\n",
|
||||||
|
error.get_msg().c_str());
|
||||||
|
notmuch->exception_reported = TRUE;
|
||||||
|
status = NOTMUCH_STATUS_XAPIAN_EXCEPTION;
|
||||||
|
}
|
||||||
|
|
||||||
|
*out = list;
|
||||||
|
|
||||||
|
DONE:
|
||||||
|
if (status && list)
|
||||||
|
talloc_free (list);
|
||||||
|
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
notmuch_bool_t
|
||||||
|
notmuch_config_list_valid (notmuch_config_list_t *metadata)
|
||||||
|
{
|
||||||
|
if (metadata->iterator == metadata->notmuch->xapian_db->metadata_keys_end ())
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *
|
||||||
|
notmuch_config_list_key (notmuch_config_list_t *list)
|
||||||
|
{
|
||||||
|
if (list->current_key)
|
||||||
|
talloc_free (list->current_key);
|
||||||
|
|
||||||
|
list->current_key = talloc_strdup (list, (*list->iterator).c_str () + CONFIG_PREFIX.length ());
|
||||||
|
|
||||||
|
return list->current_key;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *
|
||||||
|
notmuch_config_list_value (notmuch_config_list_t *list)
|
||||||
|
{
|
||||||
|
std::string strval;
|
||||||
|
notmuch_status_t status;
|
||||||
|
const char *key = notmuch_config_list_key (list);
|
||||||
|
|
||||||
|
/* TODO: better error reporting?? */
|
||||||
|
status = _metadata_value (list->notmuch, key, strval);
|
||||||
|
if (status)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
if (list->current_val)
|
||||||
|
talloc_free (list->current_val);
|
||||||
|
|
||||||
|
list->current_val = talloc_strdup (list, strval.c_str ());
|
||||||
|
return list->current_val;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
notmuch_config_list_move_to_next (notmuch_config_list_t *list)
|
||||||
|
{
|
||||||
|
list->iterator++;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
notmuch_config_list_destroy (notmuch_config_list_t *list)
|
||||||
|
{
|
||||||
|
talloc_free (list);
|
||||||
|
}
|
||||||
|
|
|
@ -206,6 +206,7 @@ typedef struct _notmuch_message notmuch_message_t;
|
||||||
typedef struct _notmuch_tags notmuch_tags_t;
|
typedef struct _notmuch_tags notmuch_tags_t;
|
||||||
typedef struct _notmuch_directory notmuch_directory_t;
|
typedef struct _notmuch_directory notmuch_directory_t;
|
||||||
typedef struct _notmuch_filenames notmuch_filenames_t;
|
typedef struct _notmuch_filenames notmuch_filenames_t;
|
||||||
|
typedef struct _notmuch_config_list notmuch_config_list_t;
|
||||||
#endif /* __DOXYGEN__ */
|
#endif /* __DOXYGEN__ */
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1858,6 +1859,49 @@ notmuch_database_set_config (notmuch_database_t *db, const char *key, const char
|
||||||
notmuch_status_t
|
notmuch_status_t
|
||||||
notmuch_database_get_config (notmuch_database_t *db, const char *key, char **value);
|
notmuch_database_get_config (notmuch_database_t *db, const char *key, char **value);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create an iterator for all config items with keys matching a given prefix
|
||||||
|
*/
|
||||||
|
notmuch_status_t
|
||||||
|
notmuch_database_get_config_list (notmuch_database_t *db, const char *prefix, notmuch_config_list_t **out);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Is 'config_list' iterator valid (i.e. _key, _value, _move_to_next can be called).
|
||||||
|
*/
|
||||||
|
notmuch_bool_t
|
||||||
|
notmuch_config_list_valid (notmuch_config_list_t *config_list);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* return key for current config pair
|
||||||
|
*
|
||||||
|
* return value is owned by the iterator, and will be destroyed by the
|
||||||
|
* next call to notmuch_config_list_key or notmuch_config_list_destroy.
|
||||||
|
*/
|
||||||
|
const char *
|
||||||
|
notmuch_config_list_key (notmuch_config_list_t *config_list);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* return 'value' for current config pair
|
||||||
|
*
|
||||||
|
* return value is owned by the iterator, and will be destroyed by the
|
||||||
|
* next call to notmuch_config_list_value or notmuch config_list_destroy
|
||||||
|
*/
|
||||||
|
const char *
|
||||||
|
notmuch_config_list_value (notmuch_config_list_t *config_list);
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* move 'config_list' iterator to the next pair
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
notmuch_config_list_move_to_next (notmuch_config_list_t *config_list);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* free any resources held by 'config_list'
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
notmuch_config_list_destroy (notmuch_config_list_t *config_list);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* interrogate the library for compile time features
|
* interrogate the library for compile time features
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -55,4 +55,64 @@ testkey2 = testvalue2
|
||||||
EOF
|
EOF
|
||||||
test_expect_equal_file EXPECTED OUTPUT
|
test_expect_equal_file EXPECTED OUTPUT
|
||||||
|
|
||||||
|
|
||||||
|
test_begin_subtest "notmuch_database_get_config_list: empty list"
|
||||||
|
cat c_head - c_tail <<'EOF' | test_C ${MAIL_DIR}
|
||||||
|
{
|
||||||
|
notmuch_config_list_t *list;
|
||||||
|
RUN(notmuch_database_get_config_list (db, "nonexistent", &list));
|
||||||
|
printf("valid = %d\n", notmuch_config_list_valid (list));
|
||||||
|
notmuch_config_list_destroy (list);
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
cat <<'EOF' >EXPECTED
|
||||||
|
== stdout ==
|
||||||
|
valid = 0
|
||||||
|
== stderr ==
|
||||||
|
EOF
|
||||||
|
test_expect_equal_file EXPECTED OUTPUT
|
||||||
|
|
||||||
|
|
||||||
|
test_begin_subtest "notmuch_database_get_config_list: all pairs"
|
||||||
|
cat c_head - c_tail <<'EOF' | test_C ${MAIL_DIR}
|
||||||
|
{
|
||||||
|
notmuch_config_list_t *list;
|
||||||
|
RUN(notmuch_database_set_config (db, "zzzafter", "afterval"));
|
||||||
|
RUN(notmuch_database_set_config (db, "aaabefore", "beforeval"));
|
||||||
|
RUN(notmuch_database_get_config_list (db, "", &list));
|
||||||
|
for (; notmuch_config_list_valid (list); notmuch_config_list_move_to_next (list)) {
|
||||||
|
printf("%s %s\n", notmuch_config_list_key (list), notmuch_config_list_value(list));
|
||||||
|
}
|
||||||
|
notmuch_config_list_destroy (list);
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
cat <<'EOF' >EXPECTED
|
||||||
|
== stdout ==
|
||||||
|
aaabefore beforeval
|
||||||
|
testkey1 testvalue1
|
||||||
|
testkey2 testvalue2
|
||||||
|
zzzafter afterval
|
||||||
|
== stderr ==
|
||||||
|
EOF
|
||||||
|
test_expect_equal_file EXPECTED OUTPUT
|
||||||
|
|
||||||
|
test_begin_subtest "notmuch_database_get_config_list: one prefix"
|
||||||
|
cat c_head - c_tail <<'EOF' | test_C ${MAIL_DIR}
|
||||||
|
{
|
||||||
|
notmuch_config_list_t *list;
|
||||||
|
RUN(notmuch_database_get_config_list (db, "testkey", &list));
|
||||||
|
for (; notmuch_config_list_valid (list); notmuch_config_list_move_to_next (list)) {
|
||||||
|
printf("%s %s\n", notmuch_config_list_key (list), notmuch_config_list_value(list));
|
||||||
|
}
|
||||||
|
notmuch_config_list_destroy (list);
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
cat <<'EOF' >EXPECTED
|
||||||
|
== stdout ==
|
||||||
|
testkey1 testvalue1
|
||||||
|
testkey2 testvalue2
|
||||||
|
== stderr ==
|
||||||
|
EOF
|
||||||
|
test_expect_equal_file EXPECTED OUTPUT
|
||||||
|
|
||||||
test_done
|
test_done
|
||||||
|
|
Loading…
Reference in a new issue