Mercurial > masqmail-0.2
diff src/masqmail.h @ 0:08114f7dcc23 0.2.21
this is masqmail-0.2.21 from oliver kurth
author | meillo@marmaro.de |
---|---|
date | Fri, 26 Sep 2008 17:05:23 +0200 |
parents | |
children | 26e34ae9a3e3 |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/masqmail.h Fri Sep 26 17:05:23 2008 +0200 @@ -0,0 +1,656 @@ +/* MasqMail + Copyright (C) 1999-2001 Oliver Kurth + + This program is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 2 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program; if not, write to the Free Software + Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. +*/ +#include <config.h> + +#include <stdio.h> +#include <stdarg.h> +#include <errno.h> +#include <stdlib.h> +#include <string.h> +#include <ctype.h> +#include <unistd.h> +#include <pwd.h> +#include <sys/types.h> +#include <sys/socket.h> +#include <netinet/in.h> +#include <time.h> +#include <sys/time.h> +#include <netinet/in.h> +#include <arpa/inet.h> +#include <netdb.h> +#include <syslog.h> +#include <signal.h> +#include <fcntl.h> + +#include <glib.h> + +#ifdef ENABLE_IDENT +#include "libident/ident.h" +#endif + +#include "lookup.h" + +typedef +struct _interface +{ + gchar *address; + gint port; +} interface; + +#define ADDR_FLAG_DELIVERED 0x01 +#define ADDR_FLAG_DEFERED 0x02 +#define ADDR_FLAG_FAILED 0x04 +#define ADDR_FLAG_LAST_ROUTE 0x40 +#define ADDR_FLAG_NOEXPAND 0x80 + +typedef struct _address +{ + gchar *address; + gchar *local_part; + gchar *domain; + gint flags; + GList *children; + struct _address *parent; +} address; + +#define addr_mark_delivered(addr) { addr->flags |= ADDR_FLAG_DELIVERED; } +#define addr_unmark_delivered(addr) { addr->flags &= ~ADDR_FLAG_DELIVERED; } +#define addr_is_delivered(addr) ((addr->flags & ADDR_FLAG_DELIVERED) != 0 ) + +#define addr_mark_defered(addr) { addr->flags |= ADDR_FLAG_DEFERED; } +#define addr_unmark_defered(addr) { addr->flags &= ~ADDR_FLAG_DEFERED; } +#define addr_is_defered(addr) ((addr->flags & ADDR_FLAG_DEFERED) != 0 ) + +#define addr_mark_failed(addr) { addr->flags |= ADDR_FLAG_FAILED; } +#define addr_unmark_failed(addr) { addr->flags &= ~ADDR_FLAG_FAILED; } +#define addr_is_failed(addr) ((addr->flags & ADDR_FLAG_FAILED) != 0 ) + +typedef +struct _connect_route +{ + gchar *name; + gchar *filename; + + gchar *protocol; + + gboolean is_local_net; + gboolean last_route; + + GList *allowed_return_paths; + GList *not_allowed_return_paths; + GList *allowed_mail_locals; + GList *not_allowed_mail_locals; + GList *allowed_rcpt_domains; + GList *not_allowed_rcpt_domains; + + interface *mail_host; + gchar *wrapper; + gboolean connect_error_fail; + + gchar *helo_name; + gboolean do_correct_helo; + gboolean do_pipelining; + + gchar *set_h_from_domain; + gchar *set_h_reply_to_domain; + gchar *set_return_path_domain; + + GList *map_h_from_addresses; + GList *map_h_reply_to_addresses; + GList *map_h_mail_followup_to_addresses; + GList *map_return_path_addresses; + + gboolean expand_h_sender_domain; + gboolean expand_h_sender_address; + + GList *resolve_list; + + gchar *auth_name; + gchar *auth_login; + gchar *auth_secret; + + gchar *pop3_login; + + gchar *pipe; + + gboolean pipe_fromline; + gboolean pipe_fromhack; +} connect_route; + +typedef struct _get_conf +{ + gchar *protocol; + gchar *server_name; + guint server_port; + gchar *wrapper; + gchar *login_user; + gchar *login_pass; + address *address; + address *return_path; + gboolean do_keep; + gboolean do_uidl; + gboolean do_uidl_dele; + gint max_size; + gboolean max_size_delete; + gint max_count; + + GList *resolve_list; + +} get_conf; + +typedef +struct _masqmail_conf +{ + gint mail_uid; + gint mail_gid; + + gint orig_uid; + gint orig_gid; + + gboolean run_as_user; + + gchar *mail_dir; + gchar *lock_dir; + gchar *spool_dir; + gchar *log_dir; + + gint debug_level; + gboolean use_syslog; + guint log_max_pri; + + gchar *host_name; + GList *local_hosts; + GList *local_addresses; + GList *not_local_addresses; + GList *local_nets; + GList *listen_addresses; + + guint remote_port; + + gboolean do_save_envelope_to; + + gboolean defer_all; + gboolean do_relay; + + GList *ident_trusted_nets; + + gboolean do_queue; + + gboolean do_verbose; + + gchar *mbox_default; + GList *mbox_users; + GList *mda_users; + GList *maildir_users; + + gchar *mda; + gboolean mda_fromline; + gboolean mda_fromhack; + + gboolean pipe_fromline; + gboolean pipe_fromhack; + + gchar *alias_file; + int (*alias_local_cmp)(const char *, const char *); + + GList *local_net_routes; + GList *connect_routes; /* list of pairs which point to lists */ + + gchar *online_detect; + gchar *online_file; + gchar *online_pipe; + interface *mserver_iface; + + GList *get_names; + GList *online_gets; /* list of pairs which point to lists */ + + gchar *errmsg_file; + gchar *warnmsg_file; + GList *warn_intervals; + gint max_defer_time; + + gchar *log_user; +} masqmail_conf; + +extern masqmail_conf conf; + +typedef +struct _table_pair +{ + gchar *key; + gpointer *value; +} table_pair; + + +typedef +enum _prot_id +{ + PROT_LOCAL = 0, + PROT_BSMTP, + PROT_SMTP, + PROT_ESMTP, + PROT_POP3, + PROT_APOP, + PROT_NUM +}prot_id; + +extern gchar *prot_names[]; + +typedef +enum _header_id +{ + HEAD_FROM = 0, + HEAD_SENDER, + HEAD_TO, + HEAD_CC, + HEAD_BCC, + HEAD_DATE, + HEAD_MESSAGE_ID, + HEAD_REPLY_TO, + HEAD_SUBJECT, + HEAD_RETURN_PATH, + HEAD_ENVELOPE_TO, + HEAD_RECEIVED, + HEAD_NUM_IDS, + HEAD_STATUS, + HEAD_UNKNOWN = HEAD_NUM_IDS, + HEAD_NONE = -1, +}header_id; + +typedef +struct _header_name +{ + gchar *header; + header_id id; +}header_name; + +typedef +struct _header +{ + header_id id; + gchar *header; + gchar *value; +}header; + + +typedef +struct _message +{ + gchar *uid; + + gchar *received_host; + prot_id received_prot; + gchar *ident; + gint transfer_id; /* for multiple messages per transfer */ + + address *return_path; + GList *rcpt_list; + GList *non_rcpt_list; + + GList *hdr_list; + GList *data_list; + + gint data_size; + time_t received_time; + time_t warned_time; + + gchar *full_sender_name; +}message; + +typedef +struct _msg_out +{ + message *msg; + + address *return_path; + GList *rcpt_list; + + GList *hdr_list; + GList *xtra_hdr_list; +}msg_out; + +typedef +struct _msgout_perhost +{ + gchar *host; + GList *msgout_list; +} msgout_perhost; + +/* flags for accept() */ +/*#define ACC_LOCAL 0x01 (we better use received_host == NULL) */ +#define ACC_HEAD_FROM_RCPT 0x01 /* create To: Header from rcpt_list (cmd line) */ +#define ACC_DEL_RCPTS 0x02 /* -t option, delete rcpts */ +#define ACC_DEL_BCC 0x04 /* -t option, delete Bcc header */ +#define ACC_RCPT_FROM_HEAD 0x08 /* -t option, get rcpts from headers */ +#define ACC_NODOT_TERM 0x10 /* a dot on a line itself does not end + the message (-oi option) */ +#define ACC_NO_RECVD_HDR 0x20 /* do not create a Received: header */ +#define ACC_MAIL_FROM_HEAD 0x40 /* get return path from header */ +#define ACC_NODOT_RELAX 0x80 /* do not be picky if message ist not terminated by a dot on a line */ +#define ACC_SAVE_ENVELOPE_TO 0x0100 /* save an existent Envelope-to header as X-Orig-Envelope-to */ + +#define DLVR_LOCAL 0x01 +#define DLVR_LAN 0x02 +#define DLVR_ONLINE 0x04 +#define DLVR_ALL (DLVR_LOCAL|DLVR_LAN|DLVR_ONLINE) + +/* transport flags */ +#define MSGSTR_FROMLINE 0x01 +#define MSGSTR_FROMHACK 0x02 + +typedef +enum _accept_error +{ + AERR_OK = 0, + AERR_TIMEOUT, + AERR_EOF, + AERR_OVERFLOW, + AERR_SYNTAX, + AERR_NOSPOOL, + AERR_NORCPT, + AERR_UNKNOWN +}accept_error; + +#define BUF_LEN 1024 +#define MAX_ADDRESS 256 +#define MAX_DATALINE 4096 + +typedef +enum _smtp_cmd_id +{ + SMTP_HELO = 0, + SMTP_EHLO, + SMTP_MAIL_FROM, + SMTP_RCPT_TO, + SMTP_DATA, + SMTP_QUIT, + SMTP_RSET, + SMTP_NOOP, + SMTP_HELP, + SMTP_NUM_IDS, + SMTP_EOF = -1, + SMTP_ERROR = -2, +} smtp_cmd_id; + +typedef +struct _smtp_cmd +{ + smtp_cmd_id id; + gchar *cmd; +} smtp_cmd; + +typedef +struct _smtp_connection +{ + gchar *remote_host; + + prot_id prot; + gint next_id; + + gboolean helo_seen; + gboolean from_seen; + gboolean rcpt_seen; + + message *msg; +}smtp_connection; + +/* alias.c*/ +gboolean addr_is_local(address *addr); +GList *alias_expand(GList *alias_table, GList *rcpt_list, GList *non_rcpt_list); + +/* child.c */ +int child(const char *command); + +/* conf.c */ +void init_conf(); +gboolean read_conf(gchar *filename); +connect_route *read_route(gchar *filename, gboolean is_local_net); +GList *read_route_list(GList *rf_list, gboolean is_local_net); +void destroy_route(connect_route *r); +void destroy_route_list(GList *list); +get_conf *read_get_conf(gchar *filename); +void destroy_get_conf(get_conf *gc); +connect_route *create_local_route(); + +/* expand.c */ +GList *var_table_rcpt(GList *var_table, address *rcpt); +GList *var_table_msg(GList *var_table, message *msg); +GList *var_table_conf(GList *var_table); +gint expand(GList *var_list, gchar *format, gchar *result, gint result_len); + +/* message.c */ +message *create_message(void); +void destroy_message(message *msg); +void destroy_msg_list(GList *msg_list); +void msg_free_data(message *msg); +gint msg_calc_size(message *msg, gboolean is_smtp); + +msg_out *create_msg_out(message *msg); +msg_out *clone_msg_out(msg_out *msgout_orig); +GList *create_msg_out_list(GList *msg_list); +void destroy_msg_out(msg_out *msgout); +void destroy_msg_out_list(GList *msgout_list); + +/* address.c */ +address *create_address(gchar *path, gboolean is_rfc821); +address *create_address_qualified(gchar *path, gboolean is_rfc821, + gchar *domain); +address *create_address_pipe(gchar *path); +void destroy_address(address *addr); +address *copy_modify_address(const address *orig, gchar *l_part, gchar *dom); +#define copy_address(addr) copy_modify_address(addr, NULL, NULL) +gboolean addr_isequal(address *addr1, address *addr2); +gboolean addr_isequal_parent(address *addr1, address *addr2); +address *addr_find_ancestor(address *addr); +gboolean addr_is_delivered_children(address *addr); +gboolean addr_is_finished_children(address *addr); +gchar *addr_string(address *addr); +gint addr_match(address *addr1, address *addr2); + +/* accept.c */ +accept_error accept_message(FILE *in, message *msg, + guint flags); +accept_error accept_message_prepare(message *msg, guint flags); + +/* header.c */ +gchar *rec_timestamp(); +GList *find_header(GList *hdr_list, header_id id, gchar *hdr_str); +void header_unfold(header *hdr); +void header_fold(header *hdr); +header *create_header(header_id id, gchar *fmt, ...); +void destroy_header(header *hdr); +header *copy_header(header *hdr); +header *get_header(gchar *line); + +/* smtp_in.c */ +void smtp_in(FILE *in, FILE *out, gchar *remote_host, gchar *ident); + +/* listen.c */ +void listen_port(GList *addr_list, gint qival, char *argv[]); + +/* parse.c */ +gboolean split_address(const gchar *path, gchar **local_part, gchar **domain, + gboolean is_rfc821); +gboolean parse_address_rfc822(gchar *string, + gchar **local_begin, gchar **local_end, + gchar **domain_begin, gchar **domain_end, + gchar **address_end); +gboolean parse_address_rfc821(gchar *string, + gchar **local_begin, gchar **local_end, + gchar **domain_begin, gchar **domain_end, + gchar **address_end); +address *_create_address(gchar *string, gchar **end, gboolean is_rfc821); +address *create_address_rfc821(gchar *string, gchar **end); +address *create_address_rfc822(gchar *string, gchar **end); +GList *addr_list_append_rfc822(GList *addr_list, gchar *string, gchar *domain); +gboolean addr_isequal(address *addr1, address *addr2); + +/* connect.c */ +mxip_addr *connect_hostlist(int *psockfd, gchar *host, guint port, + GList *addr_list); +mxip_addr *connect_resolvelist(int *psockfd, gchar *host, guint port, + GList *res_funcs); + +/* deliver.c */ +void msg_rcptlist_local(GList *rcpt_list, GList **, GList **); +gboolean deliver_local(msg_out *msgout); +gboolean deliver_msglist_host(connect_route *route, GList *msg_list, gchar *host, GList *res_list); +gboolean deliver_route_msgout_list(connect_route *route, GList *msgout_list); +gboolean deliver_route_msg_list(connect_route *route, GList *msgout_list); +gboolean deliver_finish(msg_out *msgout); +gboolean deliver_finish_list(GList *msgout_list); +gboolean deliver_msg_list(GList *msg_list, guint flags); +gboolean deliver(message *msg); + +/* fail_msg.c */ +gboolean fail_msg(message *msg, gchar *template, + GList *failed_rcpts, gchar *err_fmt, va_list args); +gboolean warn_msg(message *msg, gchar *template, + GList *failed_rcpts, gchar *err_fmt, va_list args); + +/* get.c */ +gboolean get_from_file(gchar *fname); +gboolean get_from_name(gchar *name); +gboolean get_all(void); +void get_online(void); +void get_daemon(gint gival, char *argv[]); +gboolean pop_before_smtp(gchar *fname); + +/* interface.c */ +gboolean init_sockaddr(struct sockaddr_in *name, interface *iface); +int make_server_socket(interface *iface); + +/* local.c */ +gboolean append_file(message *msg, GList *hdr_list, gchar *user); +gboolean maildir_out(message *msg, GList *hdr_list, gchar *user, guint flags); +gboolean pipe_out(message *msg, GList *hdr_list, address *rcpt, gchar *cmd, guint flags); + +/* log.c */ +gchar *ext_strerror(int err); +gboolean logopen(void); +void logclose(void); +void vlogwrite(int pri, const char *fmt, va_list args); +void logwrite(int pri, const char *fmt, ...); +void debugf(const char *fmt, ...); +void vdebugf(const char *fmt, va_list args); +void maillog(const char *fmt, ...); + +/* spool.c */ +gboolean spool_read_data(message *msg); +gboolean spool_read_data(message *msg); +message *msg_spool_read(gchar *uid, gboolean do_readdata); +gboolean spool_write(message *msg, gboolean do_writedata); +gboolean spool_lock(gchar *uid); +gboolean spool_unlock(gchar *uid); +gboolean spool_delete_all(message *msg); + +/* queue.c */ +GList *read_queue(gboolean do_readdata); +gboolean queue_run(void); +gboolean queue_run_online(void); +void queue_list(void); +gboolean queue_delete(gchar *uid); + +/* online.c */ +gchar *detect_online(); +void set_online_name(gchar *name); + +/* permissions.c */ +gboolean is_ingroup(uid_t uid, gid_t gid); +void set_euidgid(gint uid, gint gid, uid_t *old_uid, gid_t *old_gid); +void set_identity(uid_t old_uid, gchar *task_name); + +/* rewrite.c */ +gboolean set_address_header_domain(header *hdr, gchar *domain); +gboolean map_address_header(header *hdr, GList *table); + +/* route.c */ +msgout_perhost *create_msgout_perhost(gchar *host); +void destroy_msgout_perhost(msgout_perhost *mo_ph); +void rewrite_headers(msg_out *msgout, connect_route *route); +void rcptlist_with_one_of_hostlist(GList *rcpt_list, GList *host_list, + GList **, GList **); +void rcptlist_with_addr_is_local(GList *rcpt_list, + GList **p_rcpt_list, GList **p_non_rcpt_list); +gboolean route_strip_msgout(connect_route *route, msg_out *msgout); +msg_out *route_prepare_msgout(connect_route *route, msg_out *msgout); +GList *route_msgout_list(connect_route *route, GList *msgout_list); +gboolean route_is_allowed_return_path(connect_route *route, address *ret_path); +gboolean route_is_allowed_mail_local(connect_route *route, address *ret_path); +void msg_rcptlist_route(connect_route *route, GList *rcpt_list, + GList **p_rcpt_list, GList **p_non_rcpt_list); + +/* tables.c */ +table_pair *create_pair(gchar *key, gpointer value); +table_pair *create_pair_string(gchar *key, gpointer value); +table_pair *parse_table_pair(gchar *line, char delim); +gpointer *table_find_func(GList *table_list, gchar *key, int (*cmp_func)(const char *, const char *)); +gpointer *table_find(GList *table_list, gchar *key); +gpointer *table_find_case(GList *table_list, gchar *key); +gpointer *table_find_fnmatch(GList *table_list, gchar *key); +GList *table_read(gchar *fname, gchar delim); +void destroy_table(GList *table); + +/* timeival.c */ +gint time_interval(gchar *str, gint *pos); + +/* permissions.c */ +gboolean is_privileged_user(uid_t uid); + +/* other things */ + +#define foreach(list, node)\ +for((node) = g_list_first(list);\ + (node);\ + (node) = g_list_next(node)) + +#ifdef ENABLE_DEBUG +#define DEBUG(level) if(level <= conf.debug_level) +#else +/* hopefully the compiler optmizes this away... */ +#define DEBUG(level) if(0) +#endif + +#define LOG_VERBOSE 0x100 + +#ifndef HAVE_GETLINE +#define getline(buf, size, file) getdelim(buf, size, '\n', file) +#endif + +#ifndef HAVE_FDATASYNC +#define fdatasync(fd) fsync(fd) +#endif + +#ifndef CONF_DIR +#define CONF_DIR "/etc/masqmail" +#endif + +#define CONF_FILE CONF_DIR"/masqmail.conf" + +#define PIDFILEDIR "/var/run/masqmail/" + +#ifndef va_copy +#ifdef __va_copy +#define va_copy(ap1, ap2) __va_copy(ap1, ap2) +#else +#define va_copy(ap1, ap2) G_VA_COPY(ap1, ap2) +#endif +#endif + +/* *BSD needs this: */ +extern char **environ;