1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
|
#ifndef MAIL_USER_H
#define MAIL_USER_H
#include "net.h"
#include "unichar.h"
#include "mail-storage-settings.h"
#include "process-stat.h"
struct module;
struct stats;
struct fs_settings;
struct ssl_iostream_settings;
struct mail_user;
struct dict_op_settings;
struct mail_user_vfuncs {
void (*deinit)(struct mail_user *user);
void (*deinit_pre)(struct mail_user *user);
void (*stats_fill)(struct mail_user *user, struct stats *stats);
};
struct mail_user_connection_data {
struct ip_addr *local_ip, *remote_ip;
in_port_t local_port, remote_port;
bool secured:1;
bool ssl_secured:1;
};
struct mail_user {
pool_t pool;
struct mail_user_vfuncs v, *vlast;
int refcount;
struct event *event;
/* User's creator if such exists. For example for autocreated shared
mailbox users their creator is the logged in user. */
struct mail_user *creator;
/* Set if user was created via mail_storage_service. */
struct mail_storage_service_user *_service_user;
const char *username;
/* don't access the home directly. It may be set lazily. */
const char *_home;
uid_t uid;
gid_t gid;
const char *service;
const char *session_id;
struct mail_user_connection_data conn;
const char *auth_mech, *auth_token, *auth_user;
const char *const *userdb_fields;
/* Timestamp when this session was initially created. Most importantly
this stays the same after IMAP client is hibernated and restored. */
time_t session_create_time;
const struct var_expand_table *var_expand_table;
/* If non-NULL, fail the user initialization with this error.
This could be set by plugins that need to fail the initialization. */
const char *error;
const struct setting_parser_info *set_info;
const struct mail_user_settings *unexpanded_set;
struct mail_user_settings *set;
struct mail_namespace *namespaces;
struct mail_storage *storages;
struct dict_op_settings *dict_op_set;
ARRAY(const struct mail_storage_hooks *) hooks;
normalizer_func_t *default_normalizer;
/* Filled lazily by mailbox_attribute_*() when accessing attributes. */
struct dict *_attr_dict;
/* Module-specific contexts. See mail_storage_module_id. */
ARRAY(union mail_user_module_context *) module_contexts;
struct process_stat proc_stat;
/* User doesn't exist (as reported by userdb lookup when looking
up home) */
bool nonexistent:1;
/* Either home is set or there is no home for the user. */
bool home_looked_up:1;
/* User is anonymous */
bool anonymous:1;
/* This is an autocreated user (e.g. for shared namespace or
lda raw storage) */
bool autocreated:1;
/* mail_user_init() has been called */
bool initialized:1;
/* The initial namespaces have been created and
hook_mail_namespaces_created() has been called. */
bool namespaces_created:1;
/* SET_STR_VARS in user's all settings have been expanded.
This happens near the beginning of the user initialization,
so this is rarely needed to be checked. */
bool settings_expanded:1;
/* Shortcut to mail_storage_settings.mail_debug */
bool mail_debug:1;
/* If INBOX can't be opened, log an error, but only once. */
bool inbox_open_error_logged:1;
/* Fuzzy search works for this user (FTS enabled) */
bool fuzzy_search:1;
/* We're running dsync */
bool dsyncing:1;
/* Failed to create attribute dict, don't try again */
bool attr_dict_failed:1;
/* We're deinitializing the user */
bool deinitializing:1;
/* Enable administrator user commands for the user */
bool admin:1;
/* Enable all statistics gathering */
bool stats_enabled:1;
/* This session was restored (e.g. IMAP unhibernation) */
bool session_restored:1;
};
struct mail_user_module_register {
unsigned int id;
};
union mail_user_module_context {
struct mail_user_vfuncs super;
struct mail_user_module_register *reg;
};
extern struct mail_user_module_register mail_user_module_register;
extern struct auth_master_connection *mail_user_auth_master_conn;
extern const struct var_expand_func_table *mail_user_var_expand_func_table;
struct mail_user *mail_user_alloc(struct event *parent_event,
const char *username,
const struct setting_parser_info *set_info,
const struct mail_user_settings *set);
struct mail_user *
mail_user_alloc_nodup_set(struct event *parent_event,
const char *username,
const struct setting_parser_info *set_info,
const struct mail_user_settings *set);
/* Returns -1 if settings were invalid. */
int mail_user_init(struct mail_user *user, const char **error_r);
void mail_user_ref(struct mail_user *user);
void mail_user_unref(struct mail_user **user);
/* Assert that this is the last reference for the user and unref it. */
void mail_user_deinit(struct mail_user **user);
/* Duplicate a mail_user. mail_user_init() and mail_namespaces_init() need to
be called before the user is usable. */
struct mail_user *mail_user_dup(struct mail_user *user);
/* Find another user from the given user's namespaces. */
struct mail_user *mail_user_find(struct mail_user *user, const char *name);
/* Specify mail location %variable expansion data. */
void mail_user_set_vars(struct mail_user *user, const char *service,
const struct mail_user_connection_data *conn);
/* Return %variable expansion table for the user. */
const struct var_expand_table *
mail_user_var_expand_table(struct mail_user *user);
/* Specify the user's home directory. This should be called also when it's
known that the user doesn't have a home directory to avoid the internal
lookup. */
void mail_user_set_home(struct mail_user *user, const char *home);
/* Get the home directory for the user. Returns 1 if home directory looked up
successfully, 0 if there is no home directory (either user doesn't exist or
has no home directory) or -1 if lookup failed. The returned home string
is valid until the user is freed. */
int mail_user_get_home(struct mail_user *user, const char **home_r);
/* Appends path + file prefix for creating a temporary file.
The file prefix doesn't contain any uniqueness. */
void mail_user_set_get_temp_prefix(string_t *dest,
const struct mail_user_settings *set);
/* Get volatile directory from INBOX namespace if configured. Returns NULL if
none is configured. */
const char *mail_user_get_volatile_dir(struct mail_user *user);
/* Returns 1 on success, 0 if lock_secs is reached, -1 on error */
int mail_user_lock_file_create(struct mail_user *user, const char *lock_fname,
unsigned int lock_secs,
struct file_lock **lock_r, const char **error_r);
/* Returns TRUE if plugin is loaded for the user. */
bool mail_user_is_plugin_loaded(struct mail_user *user, struct module *module);
/* If name exists in plugin_envs, return its value. */
const char *mail_user_plugin_getenv(struct mail_user *user, const char *name);
bool mail_user_plugin_getenv_bool(struct mail_user *user, const char *name);
const char *mail_user_set_plugin_getenv(const struct mail_user_settings *set,
const char *name);
bool mail_user_set_plugin_getenv_bool(const struct mail_user_settings *set,
const char *name);
/* Add more namespaces to user's namespaces. The ->next pointers may be
changed, so the namespaces pointer will be updated to user->namespaces. */
void mail_user_add_namespace(struct mail_user *user,
struct mail_namespace **namespaces);
/* Drop autocreated shared namespaces that don't have any "usable" mailboxes. */
void mail_user_drop_useless_namespaces(struct mail_user *user);
/* Replace ~/ at the beginning of the path with the user's home directory. */
const char *mail_user_home_expand(struct mail_user *user, const char *path);
/* Returns 0 if ok, -1 if home directory isn't set. */
int mail_user_try_home_expand(struct mail_user *user, const char **path);
/* Returns unique user+ip identifier for anvil. */
const char *mail_user_get_anvil_userip_ident(struct mail_user *user);
/* Basically the same as mail_storage_find_class(), except automatically load
storage plugins when needed. */
struct mail_storage *
mail_user_get_storage_class(struct mail_user *user, const char *name);
/* Initialize SSL client settings from mail_user settings. */
void mail_user_init_ssl_client_settings(struct mail_user *user,
struct ssl_iostream_settings *ssl_set_r);
/* Initialize fs_settings from mail_user settings. */
void mail_user_init_fs_settings(struct mail_user *user,
struct fs_settings *fs_set,
struct ssl_iostream_settings *ssl_set_r);
/* Fill statistics for user. By default there are no statistics, so stats
plugin must be loaded to have anything filled. */
void mail_user_stats_fill(struct mail_user *user, struct stats *stats);
/* Try to mkdir() user's home directory. Ideally this should be called only
after the caller tries to create a file to the home directory, but it fails
with ENOENT. This way it avoids unnecessary disk IO to the home. */
int mail_user_home_mkdir(struct mail_user *user);
/* Return dict_op_settings for the user. The returned settings are valid until
the user is freed. */
const struct dict_op_settings *
mail_user_get_dict_op_settings(struct mail_user *user);
/* Obtain the postmaster address to be used for this user as an RFC 5322 (IMF)
address. Returns false if the configured postmaster address is invalid in
which case error_r contains the error message. */
static inline bool
mail_user_get_postmaster_address(struct mail_user *user,
const struct message_address **address_r,
const char **error_r)
{
return mail_user_set_get_postmaster_address(user->set, address_r,
error_r);
}
/* Obtain the postmaster address to be used for this user as an RFC 5321 (SMTP)
address. Returns false if the configured postmaster address is invalid in
which case error_r contains the error message. */
static inline bool
mail_user_get_postmaster_smtp(struct mail_user *user,
const struct smtp_address **address_r,
const char **error_r)
{
return mail_user_set_get_postmaster_smtp(user->set, address_r,
error_r);
}
#endif
|