Cyrus IMAP Server: Locking

Mailboxes

For mailboxes, we lock in this order:

  • Mailbox Namelock (shared) <== possibly reversible with conversationsdb

  • user conversations db

  • cyrus.index

If you want to do any mailboxes.db transactions, they need to open and close without changing any mailbox locking during the transaction.

Likewise seen and statuscache are always done without changing mailbox locking during their transaction.

Annotations databases are a mess.

Xapian

  • Xapian per-user NAMELOCK (shared or exclusive)

  • xapianactive file lock (shared or exclusive)

Shared namelock holds the following invariants:

  • xapianactive file contents are not changed

  • directories mentioned in xapianactive are not cleaned up

Xapianactive exclusive lock holds the following invariants:

  • owner may write to first database mentioned in xapianactive

Xapianactive shared lock holds:

  • all databases in xapianactive are readable and a consistent read can be made across all of them, even with multiple requests while the lock is held.

Locking orders

SHARED case:

  • user conversations db <=== possibly reversible with SHARED xapian namelock

  • SHARED xapian namelock

  • xapianactive lock (shared to search, exclusive to write)

  • cyrus.index may be locked either side of the xapianactive lock, because the conversationsdb lock protects it from races.

EXCLUSIVE case:

  • EXCLUSIVE xapian namelock That’s it. While you’ve got this, you can add or delete items from the xapianactive file, and delete paths on disk for directories that you have removed (either during or after locking). No other locks are permitted.

If you hold a SHARED xapian namelock, you may write to a .NEW folder for a xapianactive entry that you created without taking any additional locks, because nothing can clean it under you, and nothing else can read it. This is how the repack case works.

Lock lifetime

  • Shared mailbox namelock: * possibly hours

  • conversations db and below * short as possible

Mailbox namelock holds the following invariants:

  • cyrus.index may not be repacked, however flags and modseqs may be updated

  • cyrus.annotations records may change (kind of buggy and bad, ideally we’d always write new ones if we changed them and keep the old ones)

  • cyrus.cache may be appended, but never changed

  • spool files may not be deleted (already can’t be changed)