masqmail

view src/masqmail.h @ 354:08932c629849

reworked the route concept; removed the idea of the localnet Renamed to reflect the actual meaning more clearly: s/online_routes/query_routes/g s/local_net_route/permanent_routes/g Removed local_nets, which are now represented by allowed_recipients in a permanent route. (See. examples/localnet.route) There is no more abiguity between `local' and `local net'. Run admin/config-transition on your config to learn how to update it.
author markus schnalke <meillo@marmaro.de>
date Sun, 04 Sep 2011 11:25:38 +0200
parents 332999b1303f
children f4ba4ed28848
line source
1 /* MasqMail
2 Copyright (C) 1999-2001 Oliver Kurth
3 Copyright (C) 2010 markus schnalke <meillo@marmaro.de>
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 2 of the License, or
8 (at your option) any later version.
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software
17 Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
18 */
19 #include <config.h>
21 #include <stdio.h>
22 #include <stdarg.h>
23 #include <errno.h>
24 #include <stdlib.h>
25 #include <string.h>
26 #include <ctype.h>
27 #include <unistd.h>
28 #include <pwd.h>
29 #include <sys/types.h>
30 #include <sys/socket.h>
31 #include <netinet/in.h>
32 #include <time.h>
33 #include <sys/time.h>
34 #include <netinet/in.h>
35 #include <arpa/inet.h>
36 #include <netdb.h>
37 #include <syslog.h>
38 #include <signal.h>
39 #include <fcntl.h>
41 #include <glib.h>
43 #ifdef ENABLE_IDENT
44 #include "libident/ident.h"
45 #endif
47 #include "lookup.h"
49 typedef struct _interface {
50 gchar *address;
51 gint port;
52 } interface;
54 #define ADDR_FLAG_DELIVERED 0x01
55 #define ADDR_FLAG_DEFERED 0x02
56 #define ADDR_FLAG_FAILED 0x04
57 #define ADDR_FLAG_LAST_ROUTE 0x40
59 typedef struct _address {
60 gchar *address;
61 gchar *local_part;
62 gchar *domain;
63 gint flags;
64 GList *children;
65 struct _address *parent;
66 } address;
68 #define addr_mark_delivered(addr) { addr->flags |= ADDR_FLAG_DELIVERED; }
69 #define addr_unmark_delivered(addr) { addr->flags &= ~ADDR_FLAG_DELIVERED; }
70 #define addr_is_delivered(addr) ((addr->flags & ADDR_FLAG_DELIVERED) != 0 )
72 #define addr_mark_defered(addr) { addr->flags |= ADDR_FLAG_DEFERED; }
73 #define addr_unmark_defered(addr) { addr->flags &= ~ADDR_FLAG_DEFERED; }
74 #define addr_is_defered(addr) ((addr->flags & ADDR_FLAG_DEFERED) != 0 )
76 #define addr_mark_failed(addr) { addr->flags |= ADDR_FLAG_FAILED; }
77 #define addr_unmark_failed(addr) { addr->flags &= ~ADDR_FLAG_FAILED; }
78 #define addr_is_failed(addr) ((addr->flags & ADDR_FLAG_FAILED) != 0 )
80 typedef struct _connect_route {
81 gchar *name;
82 gchar *filename;
84 gboolean is_perma;
85 gboolean last_route;
87 GList *allowed_senders;
88 GList *denied_senders;
89 GList *allowed_recipients;
90 GList *denied_recipients;
92 interface *mail_host;
93 gboolean connect_error_fail;
94 GList *resolve_list;
95 gchar *helo_name;
96 gboolean do_correct_helo;
97 gboolean instant_helo;
98 gboolean do_pipelining;
99 gchar *auth_name;
100 gchar *auth_login;
101 gchar *auth_secret;
102 gchar *wrapper;
104 gchar *set_h_from_domain;
105 gchar *set_h_reply_to_domain;
106 gchar *set_return_path_domain;
107 GList *map_h_from_addresses;
108 GList *map_h_reply_to_addresses;
109 GList *map_h_mail_followup_to_addresses;
110 GList *map_return_path_addresses;
111 gboolean expand_h_sender_domain;
112 gboolean expand_h_sender_address;
114 gchar *pipe;
115 gboolean pipe_fromline;
116 gboolean pipe_fromhack;
117 } connect_route;
119 typedef struct _masqmail_conf {
120 gint mail_uid;
121 gint mail_gid;
123 gint orig_uid;
124 gint orig_gid;
126 gboolean run_as_user;
128 gchar *mail_dir;
129 gchar *lock_dir;
130 gchar *spool_dir;
131 gchar *log_dir;
133 gint debug_level;
134 gboolean use_syslog;
135 guint log_max_pri;
137 gchar *host_name;
138 GList *local_hosts;
139 GList *local_addresses;
140 GList *not_local_addresses;
141 GList *listen_addresses;
143 /* ANSI C defines unsigned long to be at least 32bit
144 i.e. ca. 4GB max; that should be enough. */
145 gulong max_msg_size;
147 gboolean do_save_envelope_to;
149 gboolean defer_all;
150 gboolean do_relay;
152 GList *ident_trusted_nets;
154 gboolean do_queue;
156 gboolean do_verbose;
158 gchar *mbox_default;
159 GList *mbox_users;
160 GList *mda_users;
162 gchar *mda;
163 gboolean mda_fromline;
164 gboolean mda_fromhack;
166 gboolean pipe_fromline;
167 gboolean pipe_fromhack;
169 gchar *alias_file;
170 int (*localpartcmp) (const char *, const char *);
172 GList *perma_routes;
173 GList *query_routes; /* list of pairs which point to lists */
175 gchar *online_query;
177 gchar *errmsg_file;
178 gchar *warnmsg_file;
179 GList *warn_intervals;
180 gint max_defer_time;
182 gchar *log_user;
183 } masqmail_conf;
185 extern masqmail_conf conf;
187 typedef struct _table_pair {
188 gchar *key;
189 gpointer *value;
190 } table_pair;
193 /* must match the contents of prot_names[] in accept.c */
194 typedef enum _prot_id {
195 PROT_LOCAL = 0,
196 PROT_SMTP,
197 PROT_ESMTP,
198 PROT_NUM
199 } prot_id;
201 extern gchar *prot_names[];
203 typedef enum _header_id {
204 HEAD_FROM = 0,
205 HEAD_SENDER,
206 HEAD_TO,
207 HEAD_CC,
208 HEAD_BCC,
209 HEAD_DATE,
210 HEAD_MESSAGE_ID,
211 HEAD_REPLY_TO,
212 HEAD_SUBJECT,
213 HEAD_RETURN_PATH,
214 HEAD_ENVELOPE_TO,
215 HEAD_RECEIVED,
216 HEAD_NUM_IDS,
217 HEAD_STATUS,
218 HEAD_UNKNOWN = HEAD_NUM_IDS,
219 HEAD_NONE = -1,
220 } header_id;
222 typedef struct _header_name {
223 gchar *header;
224 header_id id;
225 } header_name;
227 typedef struct _header {
228 header_id id;
229 gchar *header;
230 gchar *value;
231 } header;
234 typedef struct _message {
235 gchar *uid;
237 gchar *received_host;
238 prot_id received_prot;
239 gchar *ident;
240 gint transfer_id; /* for multiple messages per transfer */
242 address *return_path;
243 GList *rcpt_list;
244 GList *non_rcpt_list;
246 GList *hdr_list;
247 GList *data_list;
249 gint data_size;
250 time_t received_time;
251 time_t warned_time;
253 gchar *full_sender_name;
254 } message;
256 typedef struct _msg_out {
257 message *msg;
259 address *return_path;
260 GList *rcpt_list;
262 GList *hdr_list;
263 GList *xtra_hdr_list; /* rewritten headers */
264 } msg_out;
266 typedef struct _msgout_perhost {
267 gchar *host;
268 GList *msgout_list;
269 } msgout_perhost;
271 /* flags for accept() */
272 #define ACC_RCPT_FROM_HEAD 0x08 /* -t option, get rcpts from headers */
273 #define ACC_DOT_IGNORE 0x10 /* a dot on a line itself does not end the message (-oi option) */
274 #define ACC_MAIL_FROM_HEAD 0x40 /* get return path from header */
275 #define ACC_NODOT_RELAX 0x80 /* do not be picky if message ist not terminated by a dot on a line */
276 #define ACC_SAVE_ENVELOPE_TO 0x0100 /* save an existent Envelope-to header as X-Orig-Envelope-to */
278 #define DLVR_LOCAL 0x01
279 #define DLVR_LAN 0x02
280 #define DLVR_ONLINE 0x04
281 #define DLVR_ALL (DLVR_LOCAL|DLVR_LAN|DLVR_ONLINE)
283 /* transport flags */
284 #define MSGSTR_FROMLINE 0x01
285 #define MSGSTR_FROMHACK 0x02
287 typedef enum _accept_error {
288 AERR_OK = 0,
289 AERR_TIMEOUT,
290 AERR_EOF,
291 AERR_OVERFLOW,
292 AERR_SYNTAX,
293 AERR_NOSPOOL,
294 AERR_NORCPT,
295 AERR_SIZE, /* max msg size exeeded (SMTP SIZE) */
296 AERR_UNKNOWN
297 } accept_error;
299 #define BUF_LEN 1024
300 #define MAX_ADDRESS 256
301 #define MAX_DATALINE 4096
303 typedef enum _smtp_cmd_id {
304 SMTP_HELO = 0,
305 SMTP_EHLO,
306 SMTP_MAIL_FROM,
307 SMTP_RCPT_TO,
308 SMTP_DATA,
309 SMTP_QUIT,
310 SMTP_RSET,
311 SMTP_NOOP,
312 SMTP_HELP,
313 SMTP_NUM_IDS,
314 SMTP_EOF = -1,
315 SMTP_ERROR = -2,
316 } smtp_cmd_id;
318 typedef struct _smtp_cmd {
319 smtp_cmd_id id;
320 gchar *cmd;
321 } smtp_cmd;
323 typedef struct _smtp_connection {
324 gchar *remote_host;
326 prot_id prot;
327 gint next_id;
329 gboolean helo_seen;
330 gboolean from_seen;
331 gboolean rcpt_seen;
333 message *msg;
334 } smtp_connection;
336 /* alias.c*/
337 gboolean addr_is_local(address * addr);
338 GList *alias_expand(GList * alias_table, GList * rcpt_list, GList * non_rcpt_list);
340 /* child.c */
341 int child(const char *command);
343 /* conf.c */
344 void init_conf();
345 gboolean read_conf(gchar * filename);
346 connect_route *read_route(gchar * filename, gboolean is_perma);
347 GList *read_route_list(GList * rf_list, gboolean is_perma);
348 void destroy_route(connect_route * r);
349 void destroy_route_list(GList * list);
351 /* expand.c */
352 GList *var_table_rcpt(GList * var_table, address * rcpt);
353 GList *var_table_msg(GList * var_table, message * msg);
354 GList *var_table_conf(GList * var_table);
355 gint expand(GList * var_list, gchar * format, gchar * result, gint result_len);
357 /* message.c */
358 message *create_message(void);
359 void destroy_message(message * msg);
360 void destroy_msg_list(GList * msg_list);
361 void msg_free_data(message * msg);
362 gint msg_calc_size(message * msg, gboolean is_smtp);
364 msg_out *create_msg_out(message * msg);
365 msg_out *clone_msg_out(msg_out * msgout_orig);
366 void destroy_msg_out(msg_out * msgout);
367 void destroy_msg_out_list(GList * msgout_list);
369 /* address.c */
370 address *create_address(gchar * path, gboolean is_rfc821);
371 address *create_address_qualified(gchar * path, gboolean is_rfc821, gchar * domain);
372 address *create_address_pipe(gchar * path);
373 void destroy_address(address * addr);
374 address *copy_modify_address(const address * orig, gchar * l_part, gchar * dom);
375 #define copy_address(addr) copy_modify_address(addr, NULL, NULL)
376 gboolean addr_isequal(address * addr1, address * addr2, int (*cmpfunc) (const char*, const char*));
377 gboolean addr_isequal_parent(address * addr1, address * addr2, int (*cmpfunc) (const char*, const char*));
378 address *addr_find_ancestor(address * addr);
379 gboolean addr_is_delivered_children(address * addr);
380 gboolean addr_is_finished_children(address * addr);
381 gchar *addr_string(address * addr);
383 /* accept.c */
384 accept_error accept_message(FILE * in, message * msg, guint flags);
385 accept_error accept_message_prepare(message * msg, guint flags);
387 /* header.c */
388 gchar *rec_timestamp();
389 GList *find_header(GList * hdr_list, header_id id, gchar * hdr_str);
390 void header_unfold(header * hdr);
391 void header_fold(header * hdr, unsigned int maxlen);
392 header *create_header(header_id id, gchar * fmt, ...);
393 void destroy_header(header * hdr);
394 header *copy_header(header * hdr);
395 header *get_header(gchar * line);
397 /* smtp_in.c */
398 void smtp_in(FILE * in, FILE * out, gchar * remote_host, gchar * ident);
400 /* listen.c */
401 void listen_port(GList * addr_list, gint qival, char *argv[]);
403 /* parse.c */
404 gboolean split_address(const gchar * path, gchar ** local_part, gchar ** domain, gboolean is_rfc821);
405 gboolean parse_address_rfc822(gchar * string, gchar ** local_begin, gchar ** local_end, gchar ** domain_begin, gchar ** domain_end, gchar ** address_end);
406 gboolean parse_address_rfc821(gchar * string, gchar ** local_begin, gchar ** local_end, gchar ** domain_begin, gchar ** domain_end, gchar ** address_end);
407 address *_create_address(gchar * string, gchar ** end, gboolean is_rfc821);
408 address *create_address_rfc821(gchar * string, gchar ** end);
409 address *create_address_rfc822(gchar * string, gchar ** end);
410 GList *addr_list_append_rfc822(GList * addr_list, gchar * string, gchar * domain);
412 /* connect.c */
413 mxip_addr *connect_hostlist(int *psockfd, gchar * host, guint port, GList * addr_list);
414 mxip_addr *connect_resolvelist(int *psockfd, gchar * host, guint port, GList * res_funcs);
416 /* deliver.c */
417 void msg_rcptlist_local(GList * rcpt_list, GList **, GList **);
418 gboolean deliver_local(msg_out * msgout);
419 gboolean deliver_msglist_host(connect_route * route, GList * msg_list, gchar * host, GList * res_list);
420 gboolean deliver_route_msgout_list(connect_route * route, GList * msgout_list);
421 gboolean deliver_route_msg_list(connect_route * route, GList * msgout_list);
422 gboolean deliver_finish(msg_out * msgout);
423 gboolean deliver_msg_list(GList * msg_list, guint flags);
424 gboolean deliver(message * msg);
426 /* fail_msg.c */
427 gboolean fail_msg(message * msg, gchar * template, GList * failed_rcpts, gchar * err_fmt, va_list args);
428 gboolean warn_msg(message * msg, gchar * template, GList * failed_rcpts, gchar * err_fmt, va_list args);
430 /* interface.c */
431 gboolean init_sockaddr(struct sockaddr_in *name, interface * iface);
432 int make_server_socket(interface * iface);
434 /* local.c */
435 gboolean append_file(message * msg, GList * hdr_list, gchar * user);
436 gboolean pipe_out(message * msg, GList * hdr_list, address * rcpt, gchar * cmd, guint flags);
438 /* log.c */
439 gchar *ext_strerror(int err);
440 gboolean logopen(void);
441 void logclose(void);
442 void vlogwrite(int pri, const char *fmt, va_list args);
443 void logwrite(int pri, const char *fmt, ...);
444 void debugf(const char *fmt, ...);
445 void vdebugf(const char *fmt, va_list args);
446 void maillog(const char *fmt, ...);
448 /* spool.c */
449 gboolean spool_read_data(message * msg);
450 message *msg_spool_read(gchar * uid);
451 gboolean spool_write(message * msg, gboolean do_writedata);
452 gboolean spool_lock(gchar * uid);
453 gboolean spool_unlock(gchar * uid);
454 gboolean spool_delete_all(message * msg);
456 /* queue.c */
457 GList *read_queue(void);
458 gboolean queue_run(void);
459 gboolean queue_run_online(void);
460 void queue_list(void);
461 gboolean queue_delete(gchar * uid);
463 /* online.c */
464 gchar *online_query();
466 /* permissions.c */
467 gboolean is_ingroup(uid_t uid, gid_t gid);
468 void set_euidgid(gint uid, gint gid, uid_t * old_uid, gid_t * old_gid);
469 void set_identity(uid_t old_uid, gchar * task_name);
471 /* rewrite.c */
472 gboolean set_address_header_domain(header * hdr, gchar * domain);
473 gboolean map_address_header(header * hdr, GList * table);
475 /* route.c */
476 msgout_perhost *create_msgout_perhost(gchar * host);
477 void destroy_msgout_perhost(msgout_perhost * mo_ph);
478 void rewrite_headers(msg_out * msgout, connect_route * route);
479 void split_rcpts(GList* rcpt_list, GList* localnets, GList** rl_local, GList** rl_localnet, GList** rl_others);
480 GList* local_rcpts(GList* rcpt_list);
481 GList* remote_rcpts(GList* rcpt_list);
482 gboolean route_strip_msgout(connect_route * route, msg_out * msgout);
483 msg_out *route_prepare_msgout(connect_route * route, msg_out * msgout);
484 GList *route_msgout_list(connect_route * route, GList * msgout_list);
485 gboolean route_sender_is_allowed(connect_route * route, address * ret_path);
486 void route_split_rcpts(connect_route * route, GList * rcpt_list, GList ** p_rcpt_list, GList ** p_non_rcpt_list);
488 /* tables.c */
489 table_pair *create_pair(gchar * key, gpointer value);
490 table_pair *create_pair_string(gchar * key, gpointer value);
491 table_pair *parse_table_pair(gchar * line, char delim);
492 gpointer *table_find_func(GList * table_list, gchar * key, int (*cmp_func) (const char *, const char *));
493 gpointer *table_find(GList * table_list, gchar * key);
494 gpointer *table_find_case(GList * table_list, gchar * key);
495 gpointer *table_find_fnmatch(GList * table_list, gchar * key);
496 GList *table_read(gchar * fname, gchar delim);
497 void destroy_table(GList * table);
499 /* timeival.c */
500 gint time_interval(gchar * str);
502 /* permissions.c */
503 gboolean is_privileged_user(uid_t uid);
505 /* other things */
507 #define foreach(list, node)\
508 for((node) = g_list_first(list);\
509 (node);\
510 (node) = g_list_next(node))
512 #ifdef ENABLE_DEBUG
513 #define DEBUG(level) if(level <= conf.debug_level)
514 #else
515 /* hopefully the compiler optmizes this away... */
516 #define DEBUG(level) if(0)
517 #endif
519 #define LOG_VERBOSE 0x100
521 #ifndef HAVE_GETLINE
522 #define getline(buf, size, file) getdelim(buf, size, '\n', file)
523 #endif
525 #ifndef HAVE_FDATASYNC
526 #define fdatasync(fd) fsync(fd)
527 #endif
529 #ifndef CONF_DIR
530 #define CONF_DIR "/etc/masqmail"
531 #endif
533 #define CONF_FILE CONF_DIR"/masqmail.conf"
535 #define PIDFILEDIR "/var/run/masqmail/"
537 #ifndef va_copy
538 #ifdef __va_copy
539 #define va_copy(ap1, ap2) __va_copy(ap1, ap2)
540 #else
541 #define va_copy(ap1, ap2) G_VA_COPY(ap1, ap2)
542 #endif
543 #endif
545 /* *BSD needs this: */
546 extern char **environ;