Большое спасибо.
http://www.opennet.me/base/net/mail_copy_milter.txt.htmlКлючевые слова: mail, copy, milter, sendmail, (найти похожие документы)
From: SaneK <sanek@xxxx>
Newsgroups: email
Date: Mon, 9 Dec 2003 14:31:37 +0000 (UTC)
Subject: Использование Milter API в sendmail для дублирования исходящей почтыВведение
Во многих форумах встречаются треды с вопросом:
"Как настроить систему копирования почты?"Одним из вариантов реализации такой системы - это использование
sendmail в связке с Procmail, второй это использование дополнительного
мэйлера на базе самого sendmail.Не один из этих методов нельзя однозначно назвать хорошим, так же как
и метод который будет описан в этой статье (мнение автора).Данная статья не претендует на оригинальность и данный метод не является
"ноу-хау", но все же подробные описания двух перечисленных выше методов
имеются, а вот использование Milter API как то позабыли.Подробнее о Milter API можно почитать на http://www.milter.org
Итак приступим:
При тестировании использовалась система FreeBSD 5.1-Relese и sendmail 8.12.9.
Для реализации фильтра использовался язык Си, ниже приведен
код фильтра:----cpmail.c----
#include <sys/types.h>
#include <errno.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <fcntl.h>
#include "libmilter/mfapi.h"#define CP_FROM
//Адрес на который будет дублироваться почта
#define copy_to_addr "user@domain.ru"
//Файл содержащий e-mail адреса, исходящую
//почту с который необходимо дублировать
#define conf_file "/etc/mail/sm_copy_users"
// #define log_file "/var/log/copy_mail.log"typedef struct
{
char *e_mail;
struct copy_users *next;
}
copy_users;copy_users *cp_users;
int copy;void get_cp_users()
{
char line[256];
copy_users *tmp, *head;
FILE *file;
cp_users = malloc(sizeof(copy_users));
cp_users -> next = NULL;
head = cp_users;
if(!(file = fopen(conf_file,"r")))
{
fprintf(stderr,"Can't open config file: %s\nError: %s\n",conf_file, strerror(errno));
exit(1);
}while(fgets(line,sizeof(line),file) != NULL)
{
//Ignore '#' character
if(strncmp(line, "#", 1) == 0)
continue;
cp_users -> e_mail = (char *)malloc(strlen(line));
bzero(cp_users -> e_mail, strlen(line));
strncpy(cp_users -> e_mail, line, strlen(line) - 1);tmp = cp_users;
cp_users = malloc(sizeof(copy_users));
cp_users -> next = NULL;
(copy_users *)tmp -> next = cp_users;}
fclose(file);
cp_users = head;
}int user_in(char *e_mail)
{
int res = 0;
copy_users *tmp;tmp = cp_users;
while(tmp -> next != NULL)
{
if(strstr(e_mail, tmp -> e_mail) != NULL)
{
res = 1;
break;
}
tmp = (copy_users *)tmp -> next;
}return res;
}
sfsistat mlfi_header(SMFICTX *ctx, char *headerf, char *headerv)
{
return SMFIS_CONTINUE;
}sfsistat mlfi_envfrom(SMFICTX *ctx, char **argv)
{
#ifdef CP_FROM
if(user_in(argv[0]))
{
copy = 1;
}
else
copy = 0;
#endif
return SMFIS_CONTINUE;
}
sfsistat mlfi_close(SMFICTX *ctx)
{
return SMFIS_ACCEPT;
}sfsistat mlfi_eom(SMFICTX *ctx)
{
if(copy)
smfi_addrcpt(ctx, copy_to_addr);return SMFIS_CONTINUE;
}
struct smfiDesc copy_mail_filter =
{
"CopyOutgoingMail", //Filter name
SMFI_VERSION,
SMFIF_ADDRCPT, // This filter may add recipients
NULL, //Connection info filter
NULL, //SMTP HELO command filter
mlfi_envfrom, //Message from
NULL, //Message To
mlfi_header, //Header filter. Neeed for change header.
NULL,
NULL,
mlfi_eom, //End of message
NULL,
mlfi_close
};int main(int argc, char *argv[])
{
get_cp_users();
(void) smfi_setconn("inet:999@localhost");
if (smfi_register(copy_mail_filter) == MI_FAILURE)
{
fprintf(stderr,"smfi_register_failed\n");
exit(1);
}
return smfi_main();
}
---end of cpmail.c---Компилируется следующим образом:
#gcc -Wall -o cpmail cpmail.c /usr/lib/libmilter.a -pthreadИ запускается:
#cd dir
#./cpmail &Т.о. наша программа будет слушать 999 порт на который sendmail'ом будет
передаваться необходимая нам информация.Для того что бы это дело заработало необходимо в sendmail.mc внести
следующие изменения и пересобрать sendmail.cf:INPUT_MAIL_FILTER(`copy_mail_filter', `S=inet:999@localhost')
define(`confINPUT_MAIL_FILTERS', `copy_mail_filter')Далее перезапустить sendmail (killall -HUP sendmail).
Вот и все, дублирование исходящей почты работает.
Строку
INPUT_MAIL_FILTER(`copy_mail_filter', `S=inet:999@localhost') define(`confINPUT_MAIL_FILTERS', `copy_mail_filter')
В любое место пихать? Предполагаю, что не в любое.
И ещё сразу мысль. Как в исходящее письмо посредством SENDMAIL добавить адресата в "скрытую копию"?
Не будет проще. Хотя я и этого не умею :))))))
divert(-1)
divert(0)
include(`/usr/share/sendmail-cf/m4/cf.m4')
VERSIONID(`linux')dnl
OSTYPE(`linux')
define(`confDEF_USER_ID',``8:12'')dnl
undefine(`BITNET_RELAY')dnl
define(`confTO_CONNECT', `1m')dnl
define(`confTRY_NULL_MX_LIST',true)dnl
define(`confDONT_PROBE_INTERFACES',true)dnl
define(`PROCMAIL_MAILER_PATH',`/usr/bin/procmail')dnl
define(`ALIAS_FILE', `/etc/aliases')dnl
FEATURE(nouucp,reject)dnl
define(`confUSERDB_SPEC', `/etc/mail/userdb.db')dnl
define(`confPRIVACY_FLAGS', `authwarnings,novrfy,noexpn,restrictqrun')dnl
FEATURE(`no_default_msa',`dnl')dnl
FEATURE(`smrsh',`/usr/sbin/smrsh')dnl
FEATURE(`mailertable',`hash -o /etc/mail/mailertable.db')dnl
FEATURE(`virtusertable',`hash -o /etc/mail/virtusertable.db')dnl
FEATURE(redirect)dnl
FEATURE(always_add_domain)dnl
FEATURE(use_cw_file)dnl
FEATURE(use_ct_file)dnl
FEATURE(local_procmail,`',`procmail -t -Y -a $h -d $u')dnl
FEATURE(`access_db')dnl
FEATURE(`blacklist_recipients')dnl
EXPOSED_USER(`root')dnl
DAEMON_OPTIONS(`Port=smtp, Name=MSA, M=E')
INPUT_MAIL_FILTER(`copy_mail_filter', `S=inet:999@localhost')
define(`confINPUT_MAIL_FILTERS', `copy_mail_filter')
MAILER(smtp)dnl
MAILER(procmail)dnl
Cwh1.oberon.com
Вот мой конфиг все работает отменно
Вариант с 999 портом прокатил отменно! Спасибо!