2016-06-13 03:05:50 +02:00
|
|
|
/* message-property.cc - Properties are like tags, but (key,value) pairs.
|
|
|
|
* keys are allowed to repeat.
|
|
|
|
*
|
|
|
|
* This file is part of notmuch.
|
|
|
|
*
|
|
|
|
* Copyright © 2016 David Bremner
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
2018-04-29 10:35:30 +02:00
|
|
|
* along with this program. If not, see https://www.gnu.org/licenses/ .
|
2016-06-13 03:05:50 +02:00
|
|
|
*
|
|
|
|
* Author: David Bremner <david@tethera.net>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "notmuch-private.h"
|
|
|
|
#include "database-private.h"
|
|
|
|
#include "message-private.h"
|
|
|
|
|
|
|
|
notmuch_status_t
|
|
|
|
notmuch_message_get_property (notmuch_message_t *message, const char *key, const char **value)
|
|
|
|
{
|
|
|
|
if (! value)
|
|
|
|
return NOTMUCH_STATUS_NULL_POINTER;
|
|
|
|
|
|
|
|
*value = _notmuch_string_map_get (_notmuch_message_property_map (message), key);
|
|
|
|
|
|
|
|
return NOTMUCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-05-11 08:57:53 +02:00
|
|
|
notmuch_status_t
|
|
|
|
notmuch_message_count_properties (notmuch_message_t *message, const char *key, unsigned int *count)
|
|
|
|
{
|
|
|
|
if (! count || ! key || ! message)
|
|
|
|
return NOTMUCH_STATUS_NULL_POINTER;
|
|
|
|
|
|
|
|
notmuch_string_map_t *map;
|
2021-03-13 13:45:34 +01:00
|
|
|
|
2018-05-11 08:57:53 +02:00
|
|
|
map = _notmuch_message_property_map (message);
|
|
|
|
if (! map)
|
|
|
|
return NOTMUCH_STATUS_NULL_POINTER;
|
|
|
|
|
|
|
|
notmuch_string_map_iterator_t *matcher = _notmuch_string_map_iterator_create (map, key, true);
|
2021-03-13 13:45:34 +01:00
|
|
|
|
2018-05-11 08:57:53 +02:00
|
|
|
if (! matcher)
|
|
|
|
return NOTMUCH_STATUS_OUT_OF_MEMORY;
|
|
|
|
|
|
|
|
*count = 0;
|
|
|
|
while (_notmuch_string_map_iterator_valid (matcher)) {
|
|
|
|
(*count)++;
|
|
|
|
_notmuch_string_map_iterator_move_to_next (matcher);
|
|
|
|
}
|
|
|
|
|
|
|
|
_notmuch_string_map_iterator_destroy (matcher);
|
|
|
|
return NOTMUCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2016-06-13 03:05:50 +02:00
|
|
|
static notmuch_status_t
|
|
|
|
_notmuch_message_modify_property (notmuch_message_t *message, const char *key, const char *value,
|
2017-10-07 10:44:05 +02:00
|
|
|
bool delete_it)
|
2016-06-13 03:05:50 +02:00
|
|
|
{
|
|
|
|
notmuch_private_status_t private_status;
|
|
|
|
notmuch_status_t status;
|
|
|
|
char *term = NULL;
|
|
|
|
|
2018-05-11 08:57:52 +02:00
|
|
|
status = _notmuch_database_ensure_writable (notmuch_message_get_database (message));
|
2016-06-13 03:05:50 +02:00
|
|
|
if (status)
|
|
|
|
return status;
|
|
|
|
|
|
|
|
if (key == NULL || value == NULL)
|
|
|
|
return NOTMUCH_STATUS_NULL_POINTER;
|
|
|
|
|
2017-04-07 01:06:20 +02:00
|
|
|
if (strchr (key, '='))
|
2016-06-13 03:05:50 +02:00
|
|
|
return NOTMUCH_STATUS_ILLEGAL_ARGUMENT;
|
|
|
|
|
|
|
|
term = talloc_asprintf (message, "%s=%s", key, value);
|
|
|
|
|
|
|
|
if (delete_it)
|
|
|
|
private_status = _notmuch_message_remove_term (message, "property", term);
|
|
|
|
else
|
|
|
|
private_status = _notmuch_message_add_term (message, "property", term);
|
|
|
|
|
|
|
|
if (private_status)
|
|
|
|
return COERCE_STATUS (private_status,
|
|
|
|
"Unhandled error modifying message property");
|
|
|
|
if (! _notmuch_message_frozen (message))
|
|
|
|
_notmuch_message_sync (message);
|
|
|
|
|
|
|
|
if (term)
|
|
|
|
talloc_free (term);
|
|
|
|
|
|
|
|
return NOTMUCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
notmuch_status_t
|
|
|
|
notmuch_message_add_property (notmuch_message_t *message, const char *key, const char *value)
|
|
|
|
{
|
2017-10-07 10:44:05 +02:00
|
|
|
return _notmuch_message_modify_property (message, key, value, false);
|
2016-06-13 03:05:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
notmuch_status_t
|
|
|
|
notmuch_message_remove_property (notmuch_message_t *message, const char *key, const char *value)
|
|
|
|
{
|
2017-10-07 10:44:05 +02:00
|
|
|
return _notmuch_message_modify_property (message, key, value, true);
|
2016-06-13 03:05:50 +02:00
|
|
|
}
|
|
|
|
|
2017-10-17 21:09:59 +02:00
|
|
|
static
|
2016-06-13 03:05:50 +02:00
|
|
|
notmuch_status_t
|
2017-10-17 21:09:59 +02:00
|
|
|
_notmuch_message_remove_all_properties (notmuch_message_t *message, const char *key, bool prefix)
|
2016-06-13 03:05:50 +02:00
|
|
|
{
|
|
|
|
notmuch_status_t status;
|
2019-06-13 12:55:35 +02:00
|
|
|
const char *term_prefix;
|
2016-06-13 03:05:50 +02:00
|
|
|
|
2018-05-11 08:57:52 +02:00
|
|
|
status = _notmuch_database_ensure_writable (notmuch_message_get_database (message));
|
2016-06-13 03:05:50 +02:00
|
|
|
if (status)
|
|
|
|
return status;
|
|
|
|
|
|
|
|
_notmuch_message_invalidate_metadata (message, "property");
|
|
|
|
if (key)
|
2017-10-17 21:09:59 +02:00
|
|
|
term_prefix = talloc_asprintf (message, "%s%s%s", _find_prefix ("property"), key,
|
|
|
|
prefix ? "" : "=");
|
2016-06-13 03:05:50 +02:00
|
|
|
else
|
|
|
|
term_prefix = _find_prefix ("property");
|
|
|
|
|
|
|
|
/* XXX better error reporting ? */
|
|
|
|
_notmuch_message_remove_terms (message, term_prefix);
|
|
|
|
|
|
|
|
return NOTMUCH_STATUS_SUCCESS;
|
|
|
|
}
|
2016-06-13 03:05:52 +02:00
|
|
|
|
2017-10-17 21:09:59 +02:00
|
|
|
notmuch_status_t
|
|
|
|
notmuch_message_remove_all_properties (notmuch_message_t *message, const char *key)
|
|
|
|
{
|
|
|
|
return _notmuch_message_remove_all_properties (message, key, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
notmuch_status_t
|
|
|
|
notmuch_message_remove_all_properties_with_prefix (notmuch_message_t *message, const char *prefix)
|
|
|
|
{
|
|
|
|
return _notmuch_message_remove_all_properties (message, prefix, true);
|
|
|
|
}
|
|
|
|
|
2016-06-13 03:05:52 +02:00
|
|
|
notmuch_message_properties_t *
|
|
|
|
notmuch_message_get_properties (notmuch_message_t *message, const char *key, notmuch_bool_t exact)
|
|
|
|
{
|
|
|
|
notmuch_string_map_t *map;
|
2019-06-13 12:55:35 +02:00
|
|
|
|
2016-06-13 03:05:52 +02:00
|
|
|
map = _notmuch_message_property_map (message);
|
|
|
|
return _notmuch_string_map_iterator_create (map, key, exact);
|
|
|
|
}
|
|
|
|
|
|
|
|
notmuch_bool_t
|
|
|
|
notmuch_message_properties_valid (notmuch_message_properties_t *properties)
|
|
|
|
{
|
|
|
|
return _notmuch_string_map_iterator_valid (properties);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
notmuch_message_properties_move_to_next (notmuch_message_properties_t *properties)
|
|
|
|
{
|
|
|
|
return _notmuch_string_map_iterator_move_to_next (properties);
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
notmuch_message_properties_key (notmuch_message_properties_t *properties)
|
|
|
|
{
|
|
|
|
return _notmuch_string_map_iterator_key (properties);
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
notmuch_message_properties_value (notmuch_message_properties_t *properties)
|
|
|
|
{
|
|
|
|
return _notmuch_string_map_iterator_value (properties);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
notmuch_message_properties_destroy (notmuch_message_properties_t *properties)
|
|
|
|
{
|
|
|
|
_notmuch_string_map_iterator_destroy (properties);
|
|
|
|
}
|