summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDirk-Jan C. Binnema <djcb@djcbsoftware.nl>2025-08-27 21:42:45 +0300
committerDirk-Jan C. Binnema <djcb@djcbsoftware.nl>2025-08-28 16:52:21 +0300
commit0831ef7f0f9392cf56428eb31e79f0a7c2e9b9d6 (patch)
tree039b75e1d23f5ed9494d6841f01866f1b1261a21
parent2f865926d6b0819fd7ad1c0e9be41b4c15916f0b (diff)
mu-document: improve sexp updating
Ensure the cached message sexp gets updated in all cases where it's necessary.
-rw-r--r--lib/message/mu-document.cc68
-rw-r--r--lib/message/mu-document.hh4
-rw-r--r--lib/tests/test-mu-store-query.cc54
3 files changed, 93 insertions, 33 deletions
diff --git a/lib/message/mu-document.cc b/lib/message/mu-document.cc
index 428b946..a8f0b24 100644
--- a/lib/message/mu-document.cc
+++ b/lib/message/mu-document.cc
@@ -1,5 +1,5 @@
/*
-** Copyright (C) 2022-2023 Dirk-Jan C. Binnema <djcb@djcbsoftware.nl>
+** Copyright (C) 2022-2025 Dirk-Jan C. Binnema <djcb@djcbsoftware.nl>
**
** 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
@@ -50,18 +50,28 @@ Document::xapian_document() const
return xdoc_;
}
-template<typename SexpType> void
-Document::put_prop(const std::string& pname, SexpType&& val)
+static std::string
+propname(const Field& field)
{
- cached_sexp().put_props(pname, std::forward<SexpType>(val));
- dirty_sexp_ = true;
+ return std::string(":") + std::string{field.name};
}
template<typename SexpType> void
-Document::put_prop(const Field& field, SexpType&& val)
+Document::sexp_put_prop(const Field& field, SexpType&& val)
{
- put_prop(std::string(":") + std::string{field.name},
- std::forward<SexpType>(val));
+ if (field.include_in_sexp()) {
+ cached_sexp().put_props(propname(field), std::forward<SexpType>(val));
+ dirty_sexp_ = true;
+ }
+}
+
+void
+Document::sexp_remove_prop(const Field& field)
+{
+ if (field.include_in_sexp()) {
+ cached_sexp().del_prop(propname(field));
+ dirty_sexp_ = true;
+ }
}
static Xapian::TermGenerator
@@ -106,16 +116,13 @@ Document::add(Field::Id id, const std::string& val)
if (field.is_searchable())
add_search_term(xdoc_, field, val, options_);
- if (field.include_in_sexp())
- put_prop(field, val);
+ sexp_put_prop(field, val);
+
}
void
Document::add(Field::Id id, const std::vector<std::string>& vals)
{
- if (vals.empty())
- return;
-
const auto field{field_from_id(id)};
if (field.is_value())
xdoc_.add_value(field.value_no(), Mu::join(vals, SepaChar1));
@@ -129,7 +136,7 @@ Document::add(Field::Id id, const std::vector<std::string>& vals)
Sexp elms{};
for(auto&& val: vals)
elms.add(val);
- put_prop(field, std::move(elms));
+ sexp_put_prop(field, std::move(elms));
}
}
@@ -192,8 +199,7 @@ Document::add(Field::Id id, const Contacts& contacts)
if (!cvec.empty())
xdoc_.add_value(field.value_no(), join(cvec, SepaChar1));
- if (field.include_in_sexp())
- put_prop(field, make_contacts_sexp(contacts));
+ sexp_put_prop(field, make_contacts_sexp(contacts));
}
Contacts
@@ -225,15 +231,14 @@ Document::contacts_value(Field::Id id) const noexcept
}
void
-Document::add_extra_contacts(const std::string& propname, const Contacts& contacts)
+Document::add_extra_contacts(const std::string& prop_name, const Contacts& contacts)
{
if (!contacts.empty()) {
- put_prop(propname, make_contacts_sexp(contacts));
+ cached_sexp().put_props(prop_name, make_contacts_sexp(contacts));
dirty_sexp_ = true;
}
}
-
static Sexp
make_emacs_time_sexp(::time_t t)
{
@@ -256,12 +261,10 @@ Document::add(Field::Id id, int64_t val)
if (field.is_value())
xdoc_.add_value(field.value_no(), to_lexnum(val));
- if (field.include_in_sexp()) {
- if (field.is_time_t())
- put_prop(field, make_emacs_time_sexp(val));
- else
- put_prop(field, val);
- }
+ if (field.is_time_t())
+ sexp_put_prop(field, make_emacs_time_sexp(val));
+ else
+ sexp_put_prop(field, val);
}
int64_t
@@ -282,7 +285,7 @@ Document::add(Priority prio)
xdoc_.add_boolean_term(field.xapian_term(to_char(prio)));
if (field.include_in_sexp())
- put_prop(field, Sexp::Symbol(priority_name(prio)));
+ sexp_put_prop(field, Sexp::Symbol(priority_name(prio)));
}
Priority
@@ -308,10 +311,9 @@ Document::add(Flags flags)
});
if (field.include_in_sexp())
- put_prop(field, std::move(flaglist));
+ sexp_put_prop(field, std::move(flaglist));
}
-
Flags
Document::flags_value() const noexcept
{
@@ -324,12 +326,15 @@ Document::remove(Field::Id field_id)
const auto field{field_from_id(field_id)};
const auto pfx{field.xapian_prefix()};
+ bool updated{};
+
xapian_try([&]{
if (auto&& val{xdoc_.get_value(field.value_no())}; !val.empty()) {
// g_debug("removing value<%u>: '%s'", field.value_no(),
// val.c_str());
xdoc_.remove_value(field.value_no());
+ updated = true;
}
std::vector<std::string> kill_list;
@@ -344,13 +349,16 @@ Document::remove(Field::Id field_id)
// g_debug("removing term '%s'", term.c_str());
try {
xdoc_.remove_term(term);
+ updated = true;
} catch(const Xapian::InvalidArgumentError& xe) {
mu_critical("failed to remove '{}'", term);
}
}
});
-}
+ if (updated)
+ sexp_remove_prop(field);
+}
#ifdef BUILD_TESTS
@@ -367,8 +375,6 @@ Document::remove(Field::Id field_id)
assert_same_contact(CV1[i], CV2[i]); \
} while(0)
-
-
static const Contacts test_contacts = {{
Contact{"john@example.com", "John", Contact::Type::Bcc},
Contact{"ringo@example.com", "Ringo", Contact::Type::Bcc},
diff --git a/lib/message/mu-document.hh b/lib/message/mu-document.hh
index 9702562..d9855f0 100644
--- a/lib/message/mu-document.hh
+++ b/lib/message/mu-document.hh
@@ -240,8 +240,8 @@ public:
Flags flags_value() const noexcept;
private:
- template<typename SexpType> void put_prop(const Field& field, SexpType&& val);
- template<typename SexpType> void put_prop(const std::string& pname, SexpType&& val);
+ template<typename SexpType> void sexp_put_prop(const Field& field, SexpType&& val);
+ void sexp_remove_prop(const Field& field);
Sexp& cached_sexp() const {
if (cached_sexp_.empty())
diff --git a/lib/tests/test-mu-store-query.cc b/lib/tests/test-mu-store-query.cc
index 0baa799..e356e51 100644
--- a/lib/tests/test-mu-store-query.cc
+++ b/lib/tests/test-mu-store-query.cc
@@ -1078,6 +1078,57 @@ https://trac.xapian.org/ticket/719
}
}
+static void
+test_update_labels()
+{
+ const TestMap test_msgs = {{
+ "inbox/new/msg",
+ {
+R"(Message-Id: <abcde@foo.bar>
+From: "Foo Example" <bar@example.com>
+Date: Wed, 26 Oct 2022 11:01:54 -0700
+To: example@example.com
+Subject: testcase
+
+Boo!
+)"},
+ }};
+
+ TempDir tdir;
+ auto store{make_test_store(tdir.path(), test_msgs, {})};
+
+ {
+ const auto qr = store.run_query("label:shrike");
+ assert_valid_result(qr);
+ g_assert_true(qr->empty());
+ }
+
+ {
+ auto qr = store.run_query("subject:testcase");
+ assert_valid_result(qr);
+ g_assert_cmpuint(qr->size(), ==, 1U);
+
+ g_assert_true(!!qr->begin().message());
+ Message msg = *qr->begin().message();
+
+ g_assert_true(msg.sexp().to_string().find("shrike") == std::string::npos);
+
+ store.update_labels(msg, Labels::parse_delta_labels("+shrike"," ").value());
+
+ g_assert_true(msg.sexp().to_string().find("shrike") != std::string::npos);
+
+ store.update_labels(msg, Labels::parse_delta_labels("-shrike"," ").value());
+ g_assert_true(msg.sexp().to_string().find("shrike") == std::string::npos);
+ }
+
+ {
+ const auto qr = store.run_query("label:shrike");
+ assert_valid_result(qr);
+ g_assert_true(qr->empty());
+ }
+
+}
+
int
main(int argc, char* argv[])
{
@@ -1113,6 +1164,9 @@ main(int argc, char* argv[])
test_html);
g_test_add_func("/store/query/ngrams",
test_ngrams);
+ g_test_add_func("/store/query/update-labels",
+ test_update_labels);
+
return g_test_run();
}