Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix(imap): persist vanished messages immediately on EXAMINE commands #10036

Merged
merged 1 commit into from
Sep 23, 2024

Conversation

st3iny
Copy link
Member

@st3iny st3iny commented Aug 21, 2024

Problem

When QRESYNC is used the server will report all changed and vanished messages right away on each EXAMINE command. This is a bandwith optimization for IMAP. However, our code doesn't expect any (sync) changes to happen outside of explicit sync runs so we miss unsolicited changes when they are pushed on arbitrary IMAP commands (not syncs).

To solve this, Horde authors came up with a "cache" that will be notified if unsolicited changes are received. However, our Horde cache class stays in memory and doesn't actually reconcile changes with the main db cache which is the source of truth.

This can lead to situations were Horde informs us of vanished messages that we delete from the memory cache but don't actually propagate to the db. Because Horde already notified us of this change, it won't necessarily inform us again on the next explicit sync.

Solution

  1. Propagate delete uids to the db cache.
  2. Use the db cache as a source of truth for determining which UIDs are actually cached/known by the client. This is done lazily to reduce the performance impact.

I also added an integration test to cover the case of unsolicited changes being pushed by the server.

@st3iny st3iny self-assigned this Aug 21, 2024
@st3iny st3iny force-pushed the fix/imap/persist-deleted-messages branch from f57cc11 to 5421a93 Compare August 21, 2024 12:22
@st3iny st3iny force-pushed the fix/imap/persist-deleted-messages branch 2 times, most recently from 7dbbf6f to 44c5315 Compare August 30, 2024 19:05
@st3iny st3iny force-pushed the fix/imap/persist-deleted-messages branch from 44c5315 to 06fa8a8 Compare August 30, 2024 19:12
@st3iny st3iny added this to the v4.0.0 milestone Sep 2, 2024
@st3iny st3iny marked this pull request as ready for review September 2, 2024 05:25
@ChristophWurst ChristophWurst removed this from the v4.0.0 milestone Sep 3, 2024
} else {
// WARNING: This is very dangerous! We **will** miss changes when using QRESYNC without
Copy link
Contributor

@kesselb kesselb Sep 10, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Means: Don't use mail without redis/apcu?

We added Horde_Imap_Client_Cache_Backend_Null with #6880 to mitigate performance/memory issue with larger mailboxes. Without redis/apcu larger mailboxes still work, but you have a chance of seeing those ghost messages.

"Very dangerous" => should we disable the null cache again?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm, now I see the problem. Without a real cache we will miss unsolicited changes because Horde will only inform the null cache and don't persist changes anywhere. QRESYNC has side effects because it is used each time a mailbox is opened.

"Very dangerous" => should we disable the null cache again?

This would be the easiest solution. But perhaps we should implement a "thin" cache that only persists vanished messages (so it should only contain the logic I added in this PR to the cache class). This way we will still prevent said memory leak while still being informed about unsolicited sync changes. And we won't require OCP's cache to be available.


PS: I removed the whole else block in #10038 but perhaps this needs second thoughts. Maybe allowing the cache to be fully disabled for this one special case of the repair sync. It must be run without any cache whatsoever to ensure QRESYNC being disabled.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As per our discussion, I removed all traces of the old in-memory Horde cache. Only relevant information for QRESYNC is fetched from the db now. That would be the list of all cached uids and the sync token (to extract values of highestmodseq and uidvalidity).

Otherwise, the new cache class will behave like the built-in null cache in a sense that it won't persist any incoming changes.

@st3iny st3iny requested a review from GretaD as a code owner September 11, 2024 18:47
@st3iny st3iny force-pushed the fix/imap/persist-deleted-messages branch from 67a12d7 to c7dff93 Compare September 11, 2024 19:10
Signed-off-by: Richard Steinmetz <richard@steinmetz.cloud>
@st3iny st3iny force-pushed the fix/imap/persist-deleted-messages branch from 33cc44d to 3b577fe Compare September 23, 2024 06:49
@st3iny st3iny merged commit d42f23a into main Sep 23, 2024
34 checks passed
@st3iny st3iny deleted the fix/imap/persist-deleted-messages branch September 23, 2024 08:16
@kesselb
Copy link
Contributor

kesselb commented Sep 25, 2024

@st3iny backport?

@st3iny
Copy link
Member Author

st3iny commented Sep 30, 2024

/backport to stable4.0

@st3iny
Copy link
Member Author

st3iny commented Oct 2, 2024

/backport to stable3.7

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants