postfix milter实例

如果收件人超过3个且信件内容大于10个字节,则把来信静悄悄丢掉。

需要安装 sendmail-devel。

main.cf 配置如下:
smtpd_milters = inet:127.0.0.1:5888

hdmilter.c 如下:

# define SMFI_VERSION   2

#include <fcntl.h>
#include <getopt.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sysexits.h>
#include <termios.h>
#include <unistd.h>
#include <sys/ioctl.h>
#include <syslog.h>

#include <libmilter/mfapi.h>

struct mlfiPriv {
    int nrcpts;
    size_t bsize;
};


#define VERTEXT "HD-Filter-Version"
#define VERSION "0.1"

#define MLFIPRIV ((struct mlfiPriv *) smfi_getpriv(ctx))

extern sfsistat mlfi_cleanup(SMFICTX *, sfsistat);


sfsistat mlfi_envfrom(SMFICTX *ctx, char **envfrom) {
    struct mlfiPriv *priv;

    syslog(LOG_DAEMON|LOG_INFO, "----------------------------------------------");

    priv = (struct mlfiPriv *) malloc(sizeof *priv);
    if (!priv) {
       
        syslog(LOG_DAEMON|LOG_INFO, "envfrom: %s", "oom");
        return SMFIS_TEMPFAIL;
    }
    priv->nrcpts = 0;
    priv->bsize = 0;

   
    smfi_setpriv(ctx, priv);

   
    {
        char **env = envfrom;
        while (*env) {
            syslog(LOG_DAEMON|LOG_INFO, "envfrom: %s", *env);
            env++;
        }
    }
    return SMFIS_CONTINUE;
}

sfsistat mlfi_envrcpt(SMFICTX *ctx, char **envrcpt) {
   struct mlfiPriv *priv = MLFIPRIV;

    priv->nrcpts++;
    {
        char **env = envrcpt;
        while (*env) {
            syslog(LOG_DAEMON|LOG_INFO, "envrcpt: %s", *env);
            env++;
        }
    }
    return SMFIS_CONTINUE;
}


sfsistat mlfi_header(SMFICTX *ctx, char *headerf, char *headerv) {
    struct mlfiPriv *priv = MLFIPRIV;
    syslog(LOG_DAEMON|LOG_INFO, "header: %s => %s", headerf, headerv);
    return SMFIS_CONTINUE;
}

sfsistat mlfi_eoh(SMFICTX *ctx) {
    syslog(LOG_DAEMON|LOG_INFO, "eoh: %s", "...");
    return SMFIS_CONTINUE;
}

sfsistat mlfi_body(SMFICTX *ctx, u_char *bodyp, size_t bodylen) {
    struct mlfiPriv *priv = MLFIPRIV;
    priv->bsize += bodylen;

   
    syslog(LOG_DAEMON|LOG_INFO, "body: +len %d", bodylen);
    return SMFIS_CONTINUE;
}

sfsistat mlfi_eom(SMFICTX *ctx) {
    struct mlfiPriv *priv = MLFIPRIV;

    syslog(LOG_DAEMON|LOG_INFO, "eom: nrcpts %d bsize %d",
        priv->nrcpts, priv->bsize);

    if (priv->nrcpts > 3 && priv->bsize > 10)
        return mlfi_cleanup(ctx, SMFIS_DISCARD);
    else
        return mlfi_cleanup(ctx, SMFIS_CONTINUE);
}

sfsistat mlfi_close(SMFICTX *ctx) {
    syslog(LOG_DAEMON|LOG_INFO, "close: ...");
    return mlfi_cleanup(ctx, SMFIS_ACCEPT);
}

sfsistat mlfi_abort(SMFICTX *ctx) {
    syslog(LOG_DAEMON|LOG_INFO, "abort: ...");
    return mlfi_cleanup(ctx, SMFIS_CONTINUE);
}

sfsistat mlfi_cleanup(SMFICTX *ctx, sfsistat rc) {
    struct mlfiPriv *priv = MLFIPRIV;

    if (priv) {
        free(priv);
        smfi_setpriv(ctx, NULL);
    }

    return(rc);
}

struct smfiDesc smfilter = {
    "HDFilter",   
    SMFI_VERSION,   
    SMFIF_NONE,   
    NULL,       
    NULL,       
    mlfi_envfrom,   
    mlfi_envrcpt,   
    mlfi_header,   
    mlfi_eoh,   
    mlfi_body,   
    mlfi_eom,   
    mlfi_abort,   
    mlfi_close   
};


int main(int argc, char *argv[]) {
    int c, fd;
    int fg = 0;
    const char *args = "p:f";

   
    while ((c = getopt(argc, argv, args)) != -1) {
        switch (c) {
          case 'p':
            if (optarg == NULL || *optarg == '\0') {
                (void) fprintf(stderr, "Illegal conn: %s\n",
                           optarg);
                exit(EX_USAGE);
            }
            (void) smfi_setconn(optarg);
            break;

        case 'f':
            fg = 1;
            break;

        }
    }

    if (smfi_register(smfilter) == MI_FAILURE) {
        fprintf(stderr, "smfi_register failed\n");
        exit(EX_UNAVAILABLE);
    }

    if (fg) {
        return smfi_main();
    }

    if (fork() == 0) {
        close(STDIN_FILENO);
        close(STDOUT_FILENO);
        close(STDERR_FILENO);

        fd = open("/dev/tty", O_RDWR);
        if (fd >= 0) {
            ioctl(fd, TIOCNOTTY, 0);
            close(fd);
        }

        return smfi_main();
    }
    return 0;
}

Makefile如下:

all: hdmilter

hdmilter: hdmilter.c
    gcc -g -o hdmilter hdmilter.c /usr/lib64/libmilter.a /usr/lib64/libsm.a /usr/lib64/libsmutil.a -lpthread

clean:
    -@rm hdmilter

test:
    ./hdmilter -p inet:5888

注意priv结构体必须在 envfrom 分配,如果在helo或connect阶段分配,在close阶段释放,则可能会导致其他回调函数拿不到priv结构体,因为同一个tcp连接可能发送多个消息,helo/connect只回调一次,而 envfrom/envrcpt/.../close过程会因消息的个数而回调多次
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值