7 /* Postfix virtual domain mail delivery agent
9 /* \fBvirtual\fR [generic Postfix daemon options]
11 /* The \fBvirtual\fR(8) delivery agent is designed for virtual mail
12 /* hosting services. Originally based on the Postfix \fBlocal\fR(8)
14 /* agent, this agent looks up recipients with map lookups of their
15 /* full recipient address, instead of using hard-coded unix password
16 /* file lookups of the address local part only.
18 /* This delivery agent only delivers mail. Other features such as
19 /* mail forwarding, out-of-office notifications, etc., must be
20 /* configured via virtual_alias maps or via similar lookup mechanisms.
24 /* The mailbox location is controlled by the \fBvirtual_mailbox_base\fR
25 /* and \fBvirtual_mailbox_maps\fR configuration parameters (see below).
26 /* The \fBvirtual_mailbox_maps\fR table is indexed by the recipient
27 /* address as described under TABLE SEARCH ORDER below.
29 /* The mailbox pathname is constructed as follows:
32 /* \fB$virtual_mailbox_base/$virtual_mailbox_maps(\fIrecipient\fB)\fR
35 /* where \fIrecipient\fR is the full recipient address.
36 /* UNIX MAILBOX FORMAT
39 /* When the mailbox location does not end in \fB/\fR, the message
40 /* is delivered in UNIX mailbox format. This format stores multiple
41 /* messages in one textfile.
43 /* The \fBvirtual\fR(8) delivery agent prepends a "\fBFrom \fIsender
44 /* time_stamp\fR" envelope header to each message, prepends a
45 /* \fBDelivered-To:\fR message header with the envelope recipient
47 /* prepends an \fBX-Original-To:\fR header with the recipient address as
49 /* prepends a \fBReturn-Path:\fR message header with the
50 /* envelope sender address, prepends a \fB>\fR character to lines
51 /* beginning with "\fBFrom \fR", and appends an empty line.
53 /* The mailbox is locked for exclusive access while delivery is in
54 /* progress. In case of problems, an attempt is made to truncate the
55 /* mailbox to its original length.
56 /* QMAIL MAILDIR FORMAT
59 /* When the mailbox location ends in \fB/\fR, the message is delivered
60 /* in qmail \fBmaildir\fR format. This format stores one message per file.
62 /* The \fBvirtual\fR(8) delivery agent prepends a \fBDelivered-To:\fR
63 /* message header with the final envelope recipient address,
64 /* prepends an \fBX-Original-To:\fR header with the recipient address as
65 /* given to Postfix, and prepends a
66 /* \fBReturn-Path:\fR message header with the envelope sender address.
68 /* By definition, \fBmaildir\fR format does not require application-level
69 /* file locking during mail delivery or retrieval.
73 /* Mailbox ownership is controlled by the \fBvirtual_uid_maps\fR
74 /* and \fBvirtual_gid_maps\fR lookup tables, which are indexed
75 /* with the full recipient address. Each table provides
76 /* a string with the numerical user and group ID, respectively.
78 /* The \fBvirtual_minimum_uid\fR parameter imposes a lower bound on
79 /* numerical user ID values that may be specified in any
80 /* \fBvirtual_uid_maps\fR.
84 /* All delivery decisions are made using the full recipient
85 /* address, folded to lower case. See also the next section
86 /* for a few exceptions with optional address extensions.
90 /* Normally, a lookup table is specified as a text file that
91 /* serves as input to the \fBpostmap\fR(1) command. The result, an
92 /* indexed file in \fBdbm\fR or \fBdb\fR format, is used for fast
93 /* searching by the mail system.
95 /* The search order is as follows. The search stops
96 /* upon the first successful lookup.
98 /* When the recipient has an optional address extension the
99 /* \fIuser+extension@domain.tld\fR address is looked up first.
101 /* With Postfix versions before 2.1, the optional address extension
102 /* is always ignored.
104 /* The \fIuser@domain.tld\fR address, without address extension,
105 /* is looked up next.
107 /* Finally, the recipient \fI@domain\fR is looked up.
109 /* When the table is provided via other means such as NIS, LDAP
110 /* or SQL, the same lookups are done as for ordinary indexed files.
112 /* Alternatively, a table can be provided as a regular-expression
113 /* map where patterns are given as regular expressions. In that case,
114 /* only the full recipient address is given to the regular-expression
119 /* The \fBvirtual\fR(8) delivery agent is not security sensitive, provided
120 /* that the lookup tables with recipient user/group ID information are
121 /* adequately protected. This program is not designed to run chrooted.
123 /* The \fBvirtual\fR(8) delivery agent disallows regular expression
124 /* substitution of $1 etc. in regular expression lookup tables,
125 /* because that would open a security hole.
127 /* The \fBvirtual\fR(8) delivery agent will silently ignore requests
128 /* to use the \fBproxymap\fR(8) server. Instead it will open the
129 /* table directly. Before Postfix version 2.2, the virtual
130 /* delivery agent will terminate with a fatal error.
132 /* RFC 822 (ARPA Internet Text Messages)
134 /* Mail bounces when the recipient has no mailbox or when the
135 /* recipient is over disk quota. In all other cases, mail for
136 /* an existing recipient is deferred and a warning is logged.
138 /* Problems and transactions are logged to \fBsyslogd\fR(8).
139 /* Corrupted message files are marked so that the queue
140 /* manager can move them to the \fBcorrupt\fR queue afterwards.
142 /* Depending on the setting of the \fBnotify_classes\fR parameter,
143 /* the postmaster is notified of bounces and of other trouble.
145 /* This delivery agent supports address extensions in email
146 /* addresses and in lookup table keys, but does not propagate
147 /* address extension information to the result of table lookup.
149 /* Postfix should have lookup tables that can return multiple result
150 /* attributes. In order to avoid the inconvenience of maintaining
151 /* three tables, use an LDAP or MYSQL database.
152 /* CONFIGURATION PARAMETERS
155 /* Changes to \fBmain.cf\fR are picked up automatically, as
157 /* processes run for only a limited amount of time. Use the command
158 /* "\fBpostfix reload\fR" to speed up a change.
160 /* The text below provides only a parameter summary. See
161 /* \fBpostconf\fR(5) for more details including examples.
162 /* MAILBOX DELIVERY CONTROLS
165 /* .IP "\fBvirtual_mailbox_base (empty)\fR"
166 /* A prefix that the \fBvirtual\fR(8) delivery agent prepends to all pathname
167 /* results from $virtual_mailbox_maps table lookups.
168 /* .IP "\fBvirtual_mailbox_maps (empty)\fR"
169 /* Optional lookup tables with all valid addresses in the domains that
170 /* match $virtual_mailbox_domains.
171 /* .IP "\fBvirtual_minimum_uid (100)\fR"
172 /* The minimum user ID value that the \fBvirtual\fR(8) delivery agent accepts
173 /* as a result from $virtual_uid_maps table lookup.
174 /* .IP "\fBvirtual_uid_maps (empty)\fR"
175 /* Lookup tables with the per-recipient user ID that the \fBvirtual\fR(8)
176 /* delivery agent uses while writing to the recipient's mailbox.
177 /* .IP "\fBvirtual_gid_maps (empty)\fR"
178 /* Lookup tables with the per-recipient group ID for \fBvirtual\fR(8) mailbox
181 /* Available in Postfix version 2.0 and later:
182 /* .IP "\fBvirtual_mailbox_domains ($virtual_mailbox_maps)\fR"
183 /* Postfix is final destination for the specified list of domains;
184 /* mail is delivered via the $virtual_transport mail delivery transport.
185 /* .IP "\fBvirtual_transport (virtual)\fR"
186 /* The default mail delivery transport and next-hop destination for
187 /* final delivery to domains listed with $virtual_mailbox_domains.
189 /* Available in Postfix version 2.5.3 and later:
190 /* .IP "\fBstrict_mailbox_ownership (yes)\fR"
191 /* Defer delivery when a mailbox file is not owned by its recipient.
195 /* .IP "\fBvirtual_mailbox_lock (see 'postconf -d' output)\fR"
196 /* How to lock a UNIX-style \fBvirtual\fR(8) mailbox before attempting
198 /* .IP "\fBdeliver_lock_attempts (20)\fR"
199 /* The maximal number of attempts to acquire an exclusive lock on a
200 /* mailbox file or \fBbounce\fR(8) logfile.
201 /* .IP "\fBdeliver_lock_delay (1s)\fR"
202 /* The time between attempts to acquire an exclusive lock on a mailbox
203 /* file or \fBbounce\fR(8) logfile.
204 /* .IP "\fBstale_lock_time (500s)\fR"
205 /* The time after which a stale exclusive mailbox lockfile is removed.
206 /* RESOURCE AND RATE CONTROLS
209 /* .IP "\fBvirtual_destination_concurrency_limit ($default_destination_concurrency_limit)\fR"
210 /* The maximal number of parallel deliveries to the same destination
211 /* via the virtual message delivery transport.
212 /* .IP "\fBvirtual_destination_recipient_limit ($default_destination_recipient_limit)\fR"
213 /* The maximal number of recipients per message for the virtual
214 /* message delivery transport.
215 /* .IP "\fBvirtual_mailbox_limit (51200000)\fR"
216 /* The maximal size in bytes of an individual mailbox or maildir file,
217 /* or zero (no limit).
218 /* MISCELLANEOUS CONTROLS
221 /* .IP "\fBconfig_directory (see 'postconf -d' output)\fR"
222 /* The default location of the Postfix main.cf and master.cf
223 /* configuration files.
224 /* .IP "\fBdaemon_timeout (18000s)\fR"
225 /* How much time a Postfix daemon process may take to handle a
226 /* request before it is terminated by a built-in watchdog timer.
227 /* .IP "\fBdelay_logging_resolution_limit (2)\fR"
228 /* The maximal number of digits after the decimal point when logging
229 /* sub-second delay values.
230 /* .IP "\fBipc_timeout (3600s)\fR"
231 /* The time limit for sending or receiving information over an internal
232 /* communication channel.
233 /* .IP "\fBmax_idle (100s)\fR"
234 /* The maximum amount of time that an idle Postfix daemon process waits
235 /* for an incoming connection before terminating voluntarily.
236 /* .IP "\fBmax_use (100)\fR"
237 /* The maximal number of incoming connections that a Postfix daemon
238 /* process will service before terminating voluntarily.
239 /* .IP "\fBprocess_id (read-only)\fR"
240 /* The process ID of a Postfix command or daemon process.
241 /* .IP "\fBprocess_name (read-only)\fR"
242 /* The process name of a Postfix command or daemon process.
243 /* .IP "\fBqueue_directory (see 'postconf -d' output)\fR"
244 /* The location of the Postfix top-level queue directory.
245 /* .IP "\fBsyslog_facility (mail)\fR"
246 /* The syslog facility of Postfix logging.
247 /* .IP "\fBsyslog_name (see 'postconf -d' output)\fR"
248 /* The mail system name that is prepended to the process name in syslog
249 /* records, so that "smtpd" becomes, for example, "postfix/smtpd".
251 /* qmgr(8), queue manager
252 /* bounce(8), delivery status reports
253 /* postconf(5), configuration parameters
254 /* syslogd(8), system logging
256 /* Use "\fBpostconf readme_directory\fR" or
257 /* "\fBpostconf html_directory\fR" to locate this information.
258 /* VIRTUAL_README, domain hosting howto
262 /* The Secure Mailer license must be distributed with this software.
266 /* This delivery agent was originally based on the Postfix local delivery
267 /* agent. Modifications mainly consisted of removing code that either
268 /* was not applicable or that was not safe in this context: aliases,
269 /* ~user/.forward files, delivery to "|command" or to /file/name.
271 /* The \fBDelivered-To:\fR message header appears in the \fBqmail\fR
272 /* system by Daniel Bernstein.
274 /* The \fBmaildir\fR structure appears in the \fBqmail\fR system
275 /* by Daniel Bernstein.
278 /* IBM T.J. Watson Research
280 /* Yorktown Heights, NY 10598, USA
283 /* andrewm@connect.com.au
284 /* connect.com.au Pty. Ltd.
285 /* Level 3, 213 Miller St
286 /* North Sydney 2060, NSW, Australia
289 /* System library. */
291 #include <sys_defs.h>
294 #include <paths.h> /* XXX mail_spool_dir dependency */
297 /* Utility library. */
303 #include <set_eugid.h>
306 /* Global library. */
308 #include <mail_queue.h>
309 #include <recipient_list.h>
310 #include <deliver_request.h>
311 #include <deliver_completed.h>
312 #include <mail_params.h>
313 #include <mail_version.h>
314 #include <mail_conf.h>
315 #include <mail_params.h>
316 #include <mail_addr_find.h>
317 #include <flush_clnt.h>
319 /* Single server skeleton. */
321 #include <mail_server.h>
323 /* Application-specific. */
328 * Tunable parameters.
330 char *var_virt_mailbox_maps
;
331 char *var_virt_uid_maps
;
332 char *var_virt_gid_maps
;
333 int var_virt_minimum_uid
;
334 char *var_virt_mailbox_base
;
335 char *var_virt_mailbox_lock
;
336 int var_virt_mailbox_limit
;
337 char *var_mail_spool_dir
; /* XXX dependency fix */
338 bool var_strict_mbox_owner
;
343 MAPS
*virtual_mailbox_maps
;
344 MAPS
*virtual_uid_maps
;
345 MAPS
*virtual_gid_maps
;
350 int virtual_mbox_lock_mask
;
352 /* local_deliver - deliver message with extreme prejudice */
354 static int local_deliver(DELIVER_REQUEST
*rqst
, char *service
)
356 const char *myname
= "local_deliver";
357 RECIPIENT
*rcpt_end
= rqst
->rcpt_list
.info
+ rqst
->rcpt_list
.len
;
365 msg_info("local_deliver: %s from %s", rqst
->queue_id
, rqst
->sender
);
368 * Initialize the delivery attributes that are not recipient specific.
371 deliver_attr_init(&state
.msg_attr
);
372 state
.msg_attr
.queue_name
= rqst
->queue_name
;
373 state
.msg_attr
.queue_id
= rqst
->queue_id
;
374 state
.msg_attr
.fp
= rqst
->fp
;
375 state
.msg_attr
.offset
= rqst
->data_offset
;
376 state
.msg_attr
.sender
= rqst
->sender
;
377 state
.msg_attr
.dsn_envid
= rqst
->dsn_envid
;
378 state
.msg_attr
.dsn_ret
= rqst
->dsn_ret
;
379 state
.msg_attr
.relay
= service
;
380 state
.msg_attr
.msg_stats
= rqst
->msg_stats
;
381 RESET_USER_ATTR(usr_attr
, state
.level
);
382 state
.request
= rqst
;
385 * Iterate over each recipient named in the delivery request. When the
386 * mail delivery status for a given recipient is definite (i.e. bounced
387 * or delivered), update the message queue file and cross off the
388 * recipient. Update the per-message delivery status.
390 for (msg_stat
= 0, rcpt
= rqst
->rcpt_list
.info
; rcpt
< rcpt_end
; rcpt
++) {
391 state
.msg_attr
.rcpt
= *rcpt
;
392 rcpt_stat
= deliver_recipient(state
, usr_attr
);
393 if (rcpt_stat
== 0 && (rqst
->flags
& DEL_REQ_FLAG_SUCCESS
))
394 deliver_completed(state
.msg_attr
.fp
, rcpt
->offset
);
395 msg_stat
|= rcpt_stat
;
398 deliver_attr_free(&state
.msg_attr
);
402 /* local_service - perform service for client */
404 static void local_service(VSTREAM
*stream
, char *service
, char **argv
)
406 DELIVER_REQUEST
*request
;
410 * Sanity check. This service takes no command-line arguments.
413 msg_fatal("unexpected command-line argument: %s", argv
[0]);
416 * This routine runs whenever a client connects to the UNIX-domain socket
417 * that is dedicated to local mail delivery service. What we see below is
418 * a little protocol to (1) tell the client that we are ready, (2) read a
419 * delivery request from the client, and (3) report the completion status
422 if ((request
= deliver_request_read(stream
)) != 0) {
423 status
= local_deliver(request
, service
);
424 deliver_request_done(stream
, request
, status
);
428 /* pre_accept - see if tables have changed */
430 static void pre_accept(char *unused_name
, char **unused_argv
)
434 if ((table
= dict_changed_name()) != 0) {
435 msg_info("table %s has changed -- restarting", table
);
440 /* post_init - post-jail initialization */
442 static void post_init(char *unused_name
, char **unused_argv
)
446 * Drop privileges most of the time.
448 set_eugid(var_owner_uid
, var_owner_gid
);
451 * No case folding needed: the recipient address is case folded.
453 virtual_mailbox_maps
=
454 maps_create(VAR_VIRT_MAILBOX_MAPS
, var_virt_mailbox_maps
,
455 DICT_FLAG_LOCK
| DICT_FLAG_PARANOID
);
458 maps_create(VAR_VIRT_UID_MAPS
, var_virt_uid_maps
,
459 DICT_FLAG_LOCK
| DICT_FLAG_PARANOID
);
462 maps_create(VAR_VIRT_GID_MAPS
, var_virt_gid_maps
,
463 DICT_FLAG_LOCK
| DICT_FLAG_PARANOID
);
465 virtual_mbox_lock_mask
= mbox_lock_mask(var_virt_mailbox_lock
);
468 /* pre_init - pre-jail initialization */
470 static void pre_init(char *unused_name
, char **unused_argv
)
474 * Reset the file size limit from the message size limit to the mailbox
477 * We can't have mailbox size limit smaller than the message size limit,
478 * because that prohibits the delivery agent from updating the queue
481 if (var_virt_mailbox_limit
) {
482 if (var_virt_mailbox_limit
< var_message_limit
|| var_message_limit
== 0)
483 msg_fatal("main.cf configuration error: %s is smaller than %s",
484 VAR_VIRT_MAILBOX_LIMIT
, VAR_MESSAGE_LIMIT
);
485 set_file_limit(var_virt_mailbox_limit
);
494 MAIL_VERSION_STAMP_DECLARE
;
496 /* main - pass control to the single-threaded skeleton */
498 int main(int argc
, char **argv
)
500 static const CONFIG_INT_TABLE int_table
[] = {
501 VAR_VIRT_MINUID
, DEF_VIRT_MINUID
, &var_virt_minimum_uid
, 1, 0,
502 VAR_VIRT_MAILBOX_LIMIT
, DEF_VIRT_MAILBOX_LIMIT
, &var_virt_mailbox_limit
, 0, 0,
505 static const CONFIG_STR_TABLE str_table
[] = {
506 VAR_MAIL_SPOOL_DIR
, DEF_MAIL_SPOOL_DIR
, &var_mail_spool_dir
, 0, 0,
507 VAR_VIRT_MAILBOX_MAPS
, DEF_VIRT_MAILBOX_MAPS
, &var_virt_mailbox_maps
, 0, 0,
508 VAR_VIRT_UID_MAPS
, DEF_VIRT_UID_MAPS
, &var_virt_uid_maps
, 0, 0,
509 VAR_VIRT_GID_MAPS
, DEF_VIRT_GID_MAPS
, &var_virt_gid_maps
, 0, 0,
510 VAR_VIRT_MAILBOX_BASE
, DEF_VIRT_MAILBOX_BASE
, &var_virt_mailbox_base
, 1, 0,
511 VAR_VIRT_MAILBOX_LOCK
, DEF_VIRT_MAILBOX_LOCK
, &var_virt_mailbox_lock
, 1, 0,
514 static const CONFIG_BOOL_TABLE bool_table
[] = {
515 VAR_STRICT_MBOX_OWNER
, DEF_STRICT_MBOX_OWNER
, &var_strict_mbox_owner
,
520 * Fingerprint executables and core dumps.
522 MAIL_VERSION_STAMP_ALLOCATE
;
524 single_server_main(argc
, argv
, local_service
,
525 MAIL_SERVER_INT_TABLE
, int_table
,
526 MAIL_SERVER_STR_TABLE
, str_table
,
527 MAIL_SERVER_BOOL_TABLE
, bool_table
,
528 MAIL_SERVER_PRE_INIT
, pre_init
,
529 MAIL_SERVER_POST_INIT
, post_init
,
530 MAIL_SERVER_PRE_ACCEPT
, pre_accept
,
531 MAIL_SERVER_PRIVILEGED
,