Shared Mailboxes
Dovecot supports mailbox sharing in a single backend setup:
- Public Shared: Shared mailboxes created by administrators.
- User Shared: Users sharing their mailboxes to other users.
- Symlinking: Quick and dirty way of sharing a few mailboxes.
Sharing mailboxes when running multiple backends requires Dovecot Pro.
See Filesystem Permissions for common filesystem related permission problems. Note that these permissions only make sense when using a shared filesystem to enable sharing between users.
Public Shared Mailboxes
Public mailboxes are typically mailboxes that are visible to all users or to large user groups. They are created by defining a public namespace, under which all the shared mailboxes are.
See Namespaces for details of how namespaces are configured.
For example to create a public Maildir mailboxes, use:
# User's private mail location
mail_driver = maildir
mail_path = ~/Maildir
# When creating any namespaces, you must also have a private namespace:
namespace {
type = private
separator = /
prefix =
# use global mail_path
inbox = yes
}
namespace {
type = public
separator = /
prefix = Public/
mail_path = /var/mail/public
subscriptions = no
}
In the above example, you would then create Maildir mailboxes under the /var/mail/public/
directory.
Note that with Maildir format Dovecot uses Maildir++ layout by default for folders, where the folder names must begin with a "." or Dovecot will ignore them.
You can also optionally use the "fs" layout if you want the directory structure to look like:
/var/mail/public/
(root dir)/var/mail/public/info/
(maildir folder)/var/mail/public/company/
(maildir folder)
Per-user \Seen Flag
The recommended way to enable private flags for shared mailboxes is to create private indexes with mail_index_private_path
. This creates dovecot.index.pvt[.log]
files, which contain only the message UIDs and the private flags. Currently the list of private flags is hardcoded only to the \Seen flag.
namespace {
type = public
separator = /
prefix = Public/
mail_driver = maildir
mail_path = /var/mail/public
mail_index_private_path = ~/Maildir/public
subscriptions = no
}
Maildir: Keyword Sharing
Make sure you don't try to use per-user mail_control_path
directory. Otherwise dovecot-keywords
file doesn't get shared and keyword mapping breaks.
Subscriptions
Typically you want each user to have control over their own subscriptions for mailboxes in public namespaces. This is why you should set subscriptions=no
to the namespace. Dovecot will then use the parent namespace's subscriptions file.
If you don't otherwise have a namespace with empty prefix, create one:
namespace subscriptions {
prefix =
separator = /
subscriptions = yes
hidden = yes
list = no
alias_for = inbox # the INBOX namespace's name
mailbox_subscriptions_filename = subscriptions-shared
}
Read-Only Mailboxes
mbox
If you have a read-only directory structure containing mbox files, you'll need to store index files elsewhere:
namespace {
type = public
prefix = Public/
mail_driver = mbox
mail_path = /var/mail/public/
mail_index_path = /var/indexes/public
subscriptions = no
}
In the above example all the users would still be sharing the index files, so you might have problems with filesystem permissions. Alternatively you could place the index files under user's home directory.
Maildir
If your Maildir is read-only, the control and index files still need to be created somewhere. You can specify the path for these with the mail_control_path
and mail_index_path
settings. The path may point to a directory that is shared among all users, or to a per-user path. Note that if the Maildir has any keywords, the per-user control directory breaks the keywords since there is no dovecot-keywords
file.
When configuring multiple namespaces, the control/index path must be different for each namespace. Otherwise if namespaces have identically named mailboxes their control/index directories will conflict and cause all kinds of problems.
If you put the control files to a per-user directory, you must also put the index files to a per-user directory, otherwise you'll get errors. It is however possible to use shared control files but per-user index files, assuming you've set up permissions properly.
namespace {
type = public
separator = /
prefix = Public/
mail_driver = maildir
mail_path = /var/mail/public
mail_control_path = ~/Maildir/public
mail_index_path = ~/Maildir/public
subscriptions = no
}
namespace {
type = public
separator = /
prefix = Team/
mail_driver = maildir
mail_path = /var/mail/team
mail_control_path = ~/Maildir/team
mail_index_path = ~/Maildir/team
subscriptions = no
}
Example: Public mailboxes with ACLs
See acl plugin for more information about ACLs.
namespace {
type = public
separator = .
prefix = public.
mail_driver = maildir
mail_path = /var/mail/public
subscriptions = no
list = children
}
acl_driver = vfile
It's important that the namespace type is "public" regardless of whether you set the namespace prefix to "shared." or something else.
After this you have to place dovecot-acl
files in every mailbox/folder below /var/mail/public
with rights for that folder (e.g. user=someone lr
).
acl_sharing_map
is not relevant for public mailboxes (only for shared).
User Shared Mailboxes
To enable mailbox sharing, you'll need to create a shared namespace. See acl plugin for more information about ACL-specific settings.
# User's private mail location.
mail_driver = maildir
mail_path = ~/Maildir
# When creating any namespaces, you must also have a private namespace:
namespace {
type = private
separator = /
prefix =
# use global mail_path
inbox = yes
}
namespace {
type = shared
separator = /
prefix = shared/$user/
mail_path = %{owner_home}/Maildir
mail_index_private_path = ~/Maildir/shared/%{owner_user}
# If users have direct filesystem level access to their mails, it's safer
# to not share the index files between users:
#mail_index_path = ~/Maildir/shared/%{owner_user}
subscriptions = no
list = children
}
mail_plugins {
acl = yes
}
protocol imap {
mail_plugins {
imap_acl = yes
}
}
acl_driver = vfile
This creates a shared/
namespace under which each user's mailboxes are. If you have multiple domains and allow sharing between them, you might want to set namespace_prefix = shared/$domain/$username/
instead (although $user
works just fine too). If you don't, you might want to drop the domain part and instead use namespace_prefix = shared/$username/
.
Changed: 2.4.0 The shared namespaces now use $user
, $username
and $domain
template variables, rather than the old %%u
, %%n
and %%d
.
namespace_list = children
specifies that if no one has shared mailboxes to the user, the "shared" directory isn't listed by the LIST command. If you wish it to be visible always, you can set namespace_list = yes
.
The sharing user can be accessed with %{owner_user}
, %{owner_username}
and %{owner_domain}
variables. The sharing user's home directory can also be looked up via User Databases using %{owner_home}
variable. These can be used in Mail Location Setting. If the users' mailboxes can be found using a template, it's a bit more efficient to not use %{owner_home}
. For example:
mail_driver = maildir
mail_path = /var/mail/%{owner_domain}/%{owner_username}/Maildir
mail_index_private_path = ~/Maildir/shared/%{owner_user}
dbox
With dbox, the index files are a very important part of the mailboxes. You must not try to change mail_index_path
to a user-specific location. This will only result in mailbox corruption. (mail_index_private_path
can be used though.)
Filesystem Permissions
Dovecot assumes that it can access the other users' mailboxes. If you use multiple UNIX UIDs, you may have problems setting up the permissions so that the mailbox sharing works. Dovecot never modifies existing files' permissions. See Filesystem Permissions for more information.
Shared Mailbox Listing
With the above configuration it's possible to open shared mailboxes if you know their name, but they won't be visible in the mailbox list. This is because Dovecot has no way of knowing what users have shared mailboxes to whom. Iterating through all users and looking inside their mail directories would be horribly inefficient for more than a couple users.
To overcome this problem Dovecot needs a dictionary, which contains the list of users who have shared mailboxes and to whom they have shared. If the users aren't properly listed in this dictionary, their shared mailboxes won't be visible. Currently there's no way to automatically rebuild this dictionary, so make sure it doesn't get lost. If it does, each user having shared mailboxes must use the IMAP SETACL command (see below) to get the dictionary updated for themselves.
See acl_sharing_map
for plugin setting information.
You could use any dictionary driver, including SQL or Cassandra, but a simple flat file should work pretty well too:
acl_sharing_map {
dict file {
path = /var/lib/dovecot/db/shared-mailboxes.db
}
}
The IMAP processes must be able to write to the db/
directory. If you're using system users, you probably want to make it mode 0770 and group sharedusers
and set mail_access_groups=sharedusers
(or something similar).
If you use multiple domains and don't wish users to share their mailboxes to users in other domains, you can use separate dict files for each domain:
acl_sharing_map {
dict file {
path = /var/mail/%{user | domain}/shared-mailboxes.db
}
}
Using SQL dictionary
See dictionary for more information, especially about permission issues.
acl_sharing_map {
dict proxy {
name = acl
}
}
dict_server {
dict acl {
driver = sql
sql_driver = pgsql
pgsql localhost {
parameters {
dbname = mails
user = sqluser
password = sqlpass
}
}
dict_map shared/shared-boxes/user/$to/$from {
sql_table = user_shares
value_field dummy {
}
key_field from_user {
value = $from
}
key_field to_user {
value = $to
}
}
dict_map shared/shared-boxes/anyone/$from {
sql_table = anyone_shares
value_field dummy {
}
key_field from_user {
value = $from
}
}
}
}
CREATE TABLE user_shares (
from_user varchar(100) not null,
to_user varchar(100) not null,
dummy char(1) DEFAULT '1', -- always '1' currently
primary key (from_user, to_user)
);
COMMENT ON TABLE user_shares IS 'User from_user shares folders to user to_user.';
CREATE INDEX to_user
ON user_shares (to_user); -- because we always search for to_user
CREATE TABLE anyone_shares (
from_user varchar(100) not null,
dummy char(1) DEFAULT '1', -- always '1' currently
primary key (from_user)
);
COMMENT ON TABLE anyone_shares IS 'User from_user shares folders to anyone.';
Mailbox Sharing
You can use doveadm acl
to share mailboxes, or it can be done using IMAP SETACL command. It is the only way to update the shared mailbox list dictionary.
Below is a quick introduction to IMAP ACL commands. See RFC 4314 for more details.
MYRIGHTS <mailbox>
Returns the user's current rights to the mailbox.
GETACL <mailbox>
Returns the mailbox's all ACLs.
SETACL <mailbox> <id> [+|-]<rights>
Give <id>
the specified rights to the mailbox.
DELETEACL <mailbox> [-]<id>
Delete <id>
's ACL from the mailbox.
<id> | Description |
---|---|
anyone | Matches all users, including anonymous users. |
authenticated | Like "anyone", but doesn't match anonymous users. |
$group | Matches all users belonging to the group ($ is not part of the group name). |
$!group | See group-override in acl plugin (Dovecot-specific feature). |
user | Matches the given user. |
The $group
syntax is not a standard, but it is mentioned in RFC 4314 examples and is also understood by at least Cyrus IMAP. Having '-
' before the identifier specifies negative rights.
See acl plugin for list of <rights>
.
Sharing Mailboxes to Everyone
Note that you can also do this only for some users by using the second table "anyone_shares
". Every user listed in this table shares his folders with everyone. See also userdb: Extra Fields.
IMAP ACL examples
Let's begin with some simple example that first gives "read" and "lookup" rights, and later adds "write-seen" right:
1 SETACL Work user@domain rl
1 OK Setacl complete.
2 SETACL Work user@domain +s
2 OK Setacl complete.
3 GETACL Work
* ACL "Work" "user@domain" lrs "myself" lrwstipekxacd
3 OK Getacl completed.
Let's see how negative rights work by testing it on ourself. See how we initially have "lookup" right, but later we don't:
1 MYRIGHTS Work
* MYRIGHTS "Work" lrwstipekxacd
1 OK Myrights completed.
2 SETACL Work -myself l
2 OK Setacl complete.
3 GETACL Work
* ACL "Work" "-myself" l "user@domain" lr "myself" lrwstipekxacd
3 OK Getacl completed.
4 myrights Work
* MYRIGHTS "Work" rwstipekxacd
4 OK Myrights completed.
Troubleshooting
Make sure
$user
or$username
is specified in thenamespace_prefix
setting rather than the old%%u
or%%n
.Make sure the
mail_path
is set correctly in the namespace location.log_debug = category=mail
will help you see if Dovecot is trying to access correct paths.doveadm acl debug -u user@domain shared/user/box
can be helpful in figuring out why a mailbox can't be accessed.
Mailbox Sharing with Symlinks
It's possible to share mailboxes simply by symlinking them among user's private mailboxes.
See Filesystem Permissions for issues related to filesystem permissions.
Maildir
ln -s /home/user2/Maildir/.Work /home/user1/Maildir/.shared.user2
ln -s /home/user3/Maildir/.Work /home/user1/Maildir/.shared.user3
Now user1 has a "shared" directory containing "user2" and "user3" child mailboxes, which point to those users' "Work" mailbox.
With Maildir++ layout it's not possible to automatically share "mailbox and its children". You'll need to symlink each mailbox separately. With the "fs" layout this is possible though.
mbox
Doing the same as in the above Maildir example:
mkdir /home/user1/mail/shared
ln -s /home/user2/mail/Work /home/user1/mail/shared/user2
ln -s /home/user3/mail/Work /home/user1/mail/shared/user3
One additional problem with mbox format is the creation of dotlock files. The dotlocks would be created under user1's directory, which makes them useless. Make sure the locking works properly with only fcntl or flock locking (See mbox locking) and just disable dotlocks.
Alternatively instead of symlinking an mbox file, put the shared mailboxes inside a directory and symlink the entire directory.
Filesystem Permissions
IMAP processes need filesystem level permissions to access shared/public mailboxes. This means that:
If you use more than one UNIX UID for your mail users (e.g. you use system users), you'll need to make sure that all users can access the mailboxes on filesystem level. (acl plugin won't help you with this.)
You can remove write permissions on purpose from public namespace root directory to prevent users from creating new mailboxes under it.
Dovecot never modifies permissions for existing mail files or directories. When users share mailboxes between each others, the system must have been set up in a way that filesystem permissions don't get in the way. The easiest way to do that is to use only a single UID. Another possibility would be to use one or more groups for all the mail files that may be shared to other users belonging to the same group. For example if you host multiple domains, you might create a group for each domain and allow mailbox sharing (only) between users in the same domain.
System User UNIX Groups
There's no requirement to use UNIX groups (i.e. typically defined in /etc/group
) for anything. If you don't care about them, you can safely ignore this section.
If you use passwd authentication database, the IMAP process has access to all the UNIX groups defined for that user. You may use these groups when granting filesystem permissions.
If you wish to use UNIX groups defined in /etc/group
but don't use passwd userdb, you can still do this by returning system_groups_user
userdb: Extra Fields, which contains the UNIX user name whose groups are read from the group file.
You can also set up extra UNIX groups by listing them in mail_access_groups
. To have per-user UNIX groups, return mail_access_groups
as userdb extra field. The advantage of using this method is that only Dovecot mail processes have access to the group, but nothing else, such as user's SSH session.
For example, a simple way to set up shared mailbox access for all' system users is to make all mail dirs/files 0770/0660 mode and owned by group "sharedmail" and then set mail_access_groups = sharedmail
. Using more fine-grained groups of course leaks less mail data in case there's a security hole in Dovecot.
Permissions For New Mailboxes
When creating a new mailbox, Dovecot copies the permissions from the mailbox root directory. For example, with mbox if you have directories:
drwx--xr-x 8 user group 4096 2009-02-21 18:31 /home/user/mail/
drwxrwxrwx 2 user group 4096 2009-02-21 18:32 /home/user/mail/foo/
When creating a new foo/bar/ directory, Dovecot gives it permissions:
drwx--xr-x 2 user group 4096 2009-02-21 18:33 /home/user/mail/foo/bar/
As you can see, the file mode was copied from mail/ directory, not mail/foo/. The group is also preserved. If this causes problems (e.g. different users having different groups create mailboxes, causing permission denied errors when trying to preserve the group) you can set the setgid bit for the root directory:
chmod g+s /home/user/mail
This will cause the group to be automatically copied by the OS for all created files/directories under it, even if the user doesn't belong to the group.
Permissions For New Files in Mailboxes
When creating new files inside a mailbox, Dovecot copies the read/write permissions from the mailbox's directory. For example if you have:
drwx--xr-x 5 user group 4096 2009-02-21 18:53 /home/user/Maildir/.foo/
Dovecot creates files under it with modes:
drwx--xr-x 2 user group 4096 2009-02-21 18:54 cur/
drwx--xr-x 2 user group 4096 2009-02-21 18:54 new/
drwx--xr-x 2 user group 4096 2009-02-21 18:54 tmp/
-rw----r-- 1 user group 156 2009-02-21 18:54 dovecot.index.log
-rw----r-- 1 user group 17 2009-02-21 18:54 dovecot-uidlist
Note how the g+x gets copied to directories, but for files it's simply ignored. The group is copied the same way as explained in the previous section.
When mails are copied between Maildirs, it's usually done by hard linking. If the source and destination directory permissions are different, Dovecot create a new file and copies data the slow way so that it can assign the wanted destination permissions. The source and destination permission lookups are done only by looking at the mailbox root directories' permissions, not individual mail files. This may become a problem if the mail files' permissions aren't as Dovecot expects.
Permissions to New /domain/user Directories
If each user has different UIDs and you have /var/mail/domain/user/
style directories, you run into a bit of trouble. The problem is that the first user who creates /var/mail/domain/
will create it as 0700 mode, and other users can't create their own user/ directories under it anymore. The solution is to use a common group for the users and set /var/mail/
directory's permissions properly (group-suid is required):
chgrp dovemail /var/mail
chmod 02770 /var/mail # or perhaps 03770 for extra security
and in dovecot.conf
:
mail_driver = maildir
mail_path = /var/vmail/%{user | domain}/%{user | username}/Maildir
mail_access_groups = dovemail
The end result should look like this:
drwxrwsr-x 3 user dovemail 60 Oct 24 12:04 domain.example.com/
drwx--S--- 3 user user 60 Oct 24 12:04 domain.example.com/user/
Note that this requires that the mail_path
setting is in its explicit format with %variables. Using ~/Maildir
won't work, because Dovecot can't really know how far down it should copy the permissions from.
Permissions to New User Home Directories
When mail_path
begins with %{home}
or ~/
, its permissions are copied from the first existing parent directory if it has setgid-bit set. This isn't done when the path contains any other %variables.
Mail Delivery Agent Permissions
When using LDA, it uses all the same configuration files as IMAP/POP3, so you don't need to worry about it.
When using an external MDA to deliver to a shared mailbox, you need to make sure that the resulting files have proper permissions. For example with Procmail + Maildir, set UMASK=007
in .procmailrc
to make the delivered mail files group-readable. To get the file to use the proper group, set the group to the Maildir's tmp/
directory and also set its setgid bit (chmod g+s
).
Dictionary Files
Created dictionary files (e.g. acl_shared_dict = file:...
) also base their initial permissions on parent directory's permissions. After the initial creation, the permissions are permanently preserved. So if you want to use different permissions, just chown/chmod the file.