]> Sergey Matveev's repositories - public-inbox.git/blob - script/public-inbox-mda
remove Email::Address dependency
[public-inbox.git] / script / public-inbox-mda
1 #!/usr/bin/perl -w
2 # Copyright (C) 2013-2015 all contributors <meta@public-inbox.org>
3 # License: AGPLv3 or later (https://www.gnu.org/licenses/agpl-3.0.txt)
4 #
5 # Mail delivery agent for public-inbox, run from your MTA upon mail delivery
6 use strict;
7 use warnings;
8 my $usage = 'public-inbox-mda < rfc2822_message';
9
10 use Email::Filter;
11 use Email::MIME;
12 use File::Path::Expand qw/expand_filename/;
13 use IPC::Run qw(run);
14 use PublicInbox::MDA;
15 use PublicInbox::Filter;
16 use PublicInbox::Config;
17 use PublicInbox::Import;
18 use PublicInbox::Git;
19
20 # n.b: hopefully we can setup the emergency path without bailing due to
21 # user error, we really want to setup the emergency destination ASAP
22 # in case there's bugs in our code or user error.
23 my $emergency = $ENV{PI_EMERGENCY} || '~/.public-inbox/emergency/';
24 $emergency = expand_filename($emergency);
25
26 # this reads the message from stdin
27 my $filter = Email::Filter->new(emergency => $emergency);
28 my $config = PublicInbox::Config->new;
29
30 my $recipient = $ENV{ORIGINAL_RECIPIENT};
31 defined $recipient or die "ORIGINAL_RECIPIENT not defined in ENV\n";
32 my $dst = $config->lookup($recipient); # first check
33 defined $dst or exit(1);
34 my $main_repo = $dst->{mainrepo} or exit(1);
35 my $filtered; # string dest
36
37 if (PublicInbox::MDA->precheck($filter, $dst->{address}) &&
38     do_spamc($filter->simple, \$filtered)) {
39         # update our message with SA headers (in case our filter rejects it)
40         my $msg = Email::MIME->new(\$filtered);
41         $filtered = undef;
42         $filter->simple($msg);
43
44         my $filter_arg;
45         my $fcfg = $dst->{filter};
46         if (!defined $fcfg || $filter eq 'reject') {
47                 $filter_arg = $filter;
48         } elsif ($fcfg eq 'scrub') {
49                 $filter_arg = undef; # the default for legacy versions
50         } else {
51                 warn "publicinbox.$dst->{name}.filter=$fcfg invalid\n";
52                 warn "must be either 'scrub' or 'reject' (the default)\n";
53         }
54
55         if (PublicInbox::Filter->run($msg, $filter_arg)) {
56                 # run spamc again on the HTML-free message
57                 if (do_spamc($msg, \$filtered)) {
58                         $msg = Email::MIME->new(\$filtered);
59                         PublicInbox::MDA->set_list_headers($msg, $dst);
60                         $filter->simple($msg);
61
62                         END {
63                                 index_sync($main_repo) if ($? == 0);
64                         };
65                         my $git = PublicInbox::Git->new($main_repo);
66                         my $im = PublicInbox::Import->new($git,
67                                                 $dst->{name}, $recipient);
68                         if (defined $im->add($msg)) {
69                                 $im->done;
70                                 $filter->ignore; # exits
71                         }
72                         # this message is similar to what ssoma-mda shows:
73                         print STDERR "CONFLICT: Message-ID: ",
74                                 $msg->header_obj->header_raw('Message-ID'),
75                                 " exists\n";
76                 }
77         }
78 } else {
79         # Ensure emergency spam gets spamassassin headers.
80         # This makes it easier to prioritize obvious spam from less obvious
81         if (defined($filtered) && $filtered ne '') {
82                 my $drop = Email::MIME->new(\$filtered);
83                 $filtered = undef;
84                 $filter->simple($drop);
85         }
86 }
87 exit 0; # goes to emergency
88
89 # we depend on "report_safe 0" in /etc/spamassassin/*.cf with --headers
90 # not using Email::Filter->pipe here since we want the stdout of
91 # the command even on failure (spamc will set $? on error).
92 sub do_spamc {
93         my ($msg, $out) = @_;
94         eval {
95                 my $orig = $msg->as_string;
96                 run([qw/spamc -E --headers/], \$orig, $out);
97         };
98
99         return ($@ || $? || !defined($$out) || $$out eq '') ? 0 : 1;
100 }
101
102 sub index_sync {
103         my ($git_dir) = @_;
104         eval {
105                 require PublicInbox::SearchIdx;
106                 PublicInbox::SearchIdx->new($git_dir, 2)->index_sync;
107         };
108 }