diff options
| author | Dirk-Jan C. Binnema <djcb@djcbsoftware.nl> | 2025-08-27 21:42:45 +0300 |
|---|---|---|
| committer | Dirk-Jan C. Binnema <djcb@djcbsoftware.nl> | 2025-08-28 16:52:21 +0300 |
| commit | 0831ef7f0f9392cf56428eb31e79f0a7c2e9b9d6 (patch) | |
| tree | 039b75e1d23f5ed9494d6841f01866f1b1261a21 /lib | |
| parent | 2f865926d6b0819fd7ad1c0e9be41b4c15916f0b (diff) | |
mu-document: improve sexp updating
Ensure the cached message sexp gets updated in all cases where it's necessary.
Diffstat (limited to 'lib')
| -rw-r--r-- | lib/message/mu-document.cc | 68 | ||||
| -rw-r--r-- | lib/message/mu-document.hh | 4 | ||||
| -rw-r--r-- | lib/tests/test-mu-store-query.cc | 54 |
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(); } |
