--- nfo/perl/scripts/dispatchmail/bin/dispatchmail 2003/01/22 05:07:56 1.1 +++ nfo/perl/scripts/dispatchmail/bin/dispatchmail 2003/01/22 17:50:56 1.4 @@ -1,255 +1,63 @@ #!/usr/bin/perl -# ========================================================================== +# ============================================================ # -# recieveMail v0.04 -# a simple mail filter done in perl with CPAN-module "Mail::Audit" +# dispatchmail v0.05 +# A simple mail filter done in perl +# with the Perl-module "Mail::Audit" +# available from CPAN. # +# $Id: dispatchmail,v 1.4 2003/01/22 17:50:56 root Exp $ # -# 2002-11-09, joko@netfrag.org -# + recieveMail now can run globally (/etc/mail/smrsh!) -# + $HOME is taken from $ENV{HOME} or $ENV{PWD} -# + rules are taken from $HOME/Mail/.rules.pm -# + fallback mechanism(s) -# + bugfixes -# -# 2002-10-14, joko@netfrag.org -# + $LOGFILE is used now (recievemail.log) -# + tracing (uses $LOGFILE) -# + checks delivery path for existance, -# changes $LOGFILE if needed -# -# 2002-07-17, joko@netfrag.org -# + added filtering by target (destination-routing) -# (looks in "to", "cc" and "bcc") -# -# 2001-12-05, joko@netfrag.org -# + initial internal release -# -# -# TODO: -# - more sophisticated filtering -# - configuration-comfort (use perl-arrays and -hashes for rule-declaration) -# - Html-Gui to add/edit/remove rules -# - rule base located in LDAP (local delivery routing) -# - completely hide away regex-stuff and provide simpler wildcarding -# - hide needed quoting of dots (.) and ats (@) in addresses -# - provide: beginsWith(string), endsWith(string), beginsAt(string, pos|regex) -# - this could become a CPAN-module sometimes (?): -# - "String"-Object to be inherited from gives these methods to you -# - examples: -# - routeTo("mbox:/path/to/mbox") if $to->beginsWith("hello"); -# - routeTo("fax:+4930123456") if $subject->contains("gatefax"); -# - metadata: -# - add some info about the context we are running in: -# - console -# - sendmail/normal -# - sendmail/smrsh -# - add some info about the user we are doing this for: -# - username -# - home-directory -# -# WISHLIST: -# - format basic log-output similar to procmail.log -# - introduce some "extended logging" including the chosen routing path +# ============================================================ +# $Log: dispatchmail,v $ +# Revision 1.4 2003/01/22 17:50:56 root +# - refactored most code to Mail::Audit::Dispath # -# ========================================================================== +# Revision 1.3 2003/01/22 07:16:51 root +# + sub traceEntry +# + further refactorings +# +# Revision 1.2 2003/01/22 05:38:44 collector +# + prepared refactoring to 'dispatchmail' +# +# ============================================================ use strict; # don't use warnings; -use Mail::Audit; use Data::Dumper; use Getopt::Long; +use Hash::Merge qw( merge ); - -# - - - - - - - - - - - - - - - - - - - - -# options -# - - - - - - - - - - - - - - - - - - - - -my $opt_base = ''; -my $opt_rulesfile = ''; -GetOptions( - 'base=s' => \$opt_base, - 'rules=s' => \$opt_rulesfile, -); -$opt_base ||= $ENV{HOME}; -$opt_base ||= $ENV{PWD}; - - -# - - - - - - - - - - - - - - - - - - - - -# -# targets -# -# declare and initialize some variables -# these are mostly base paths -# mail should be delivered to -# -# - - - - - - - - - - - - - - - - - - - - -#my $HOME = "/home/joko/virtual/joko_mail"; -my $USER = $ENV{USER}; -my $HOME = $opt_base; -my $MAILDIR = "$HOME/Mail"; -my $LOGFILE = "$MAILDIR/recievemail.log"; -my $RULESFILE = "$MAILDIR/.rules.pm"; -my $LOCKFILE = "$HOME/.procmail.lockfile"; -my $DEFAULT = "$MAILDIR/Inbox"; -my $DEBUG = 0; -my $TRACE = 1; -my $VERBOSE = 1; - -# override settings - # override $RULESFILE if given as option on the command line - $RULESFILE = $opt_rulesfile if ($opt_rulesfile); - # change logfile - $LOGFILE = "recievemail-emerg.log" if (! -e $MAILDIR); +use lib qw( /data/libs/nfo/perl/libs ); +use Mail::Audit::Dispatch; # - - - - - - - - - - - - - - - - - - - - -# main +# options # - - - - - - - - - - - - - - - - - - - - - # "jump" into processing of new incoming mail and get a "handler" to this mail - my $incoming = Mail::Audit->new; - - # 0.a. pre flight tracing - if ($TRACE) { - s2f("Mail from " . gchomp($incoming->from) . " to " . gchomp($incoming->to)); - s2f("Subject: " . gchomp($incoming->subject)); - } - - # 0.b. pre flight checks - - # TODO: check if $HOME is empty - - # check if $HOME exists - if (! -e $MAILDIR) { - my $msg = "delivery failed, base directory $MAILDIR does not exist"; - report($msg); - accept_spool(); - } - - # 1. include rules or fallback - # check if $RULESFILE exists - if (-f $RULESFILE) { - report("loading rules from \"$RULESFILE\""); - require $RULESFILE; - } else { - #die("$RULESFILE doesn't exist"); - report("configured rulesfile \"$RULESFILE\" doesn't exist"); - accept_spool(); - } - - # 2. export required stuff to rules namespace - export_symbols(); - - # 3. run dispatcher - report("running \"rules::dispatch\""); - rules::dispatch(); - - # 4. dispatcher didn't do anything - report("dispatcher could not apply any filter, using default delivery"); - - # the default-handler: simply accept all mails and route them to "/var/spool/mail" - # $incoming->accept(); - - # if you want to reject all mails coming through to here, do a ... - # $incoming->reject; - - # catch all mails and route them to a "DEFAULT"-inbox - jaccept($DEFAULT); - - - +my $defaults = { + LOG => 1, # writes reports to logfile + VERBOSE => 1, # writes reports to STDOUT + TRACE => 1, # writes contents of messages to logfile +}; +my $args; +GetOptions( + 'user=s' => \$args->{user}, + 'base=s' => \$args->{base}, + 'rules=s' => \$args->{rules}, +); - # - - - - - - - - - - - - - - - - - - - - - # tracing & reporting - # - - - - - - - - - - - - - - - - - - - - - sub s2f { - my $str = shift; - open(FH, '>>' . $LOGFILE); - print FH $str, "\n"; - close(FH); - } - - sub report { - my $msg = shift; - # TODO: tracing, debugging - - print $msg, "\n" if $VERBOSE; - if ($TRACE) { - s2f($msg); - } - - } - - # - - - - - - - - - - - - - - - - - - - - - # processing mail - # - - - - - - - - - - - - - - - - - - - - - - sub compareTarget { - my $pattern = shift; - my $ok = 0; - $ok = 1 if ($incoming->to =~ m/$pattern/); - $ok = 1 if ($incoming->cc =~ m/$pattern/); - $ok = 1 if ($incoming->bcc =~ m/$pattern/); - return $ok; - } - - sub jaccept { - my $deliver_to = shift; - - report("delivering to: $deliver_to"); - - # check deliver_to path - if (! -e $deliver_to) { - report("deliver_to path \"$deliver_to\" doesn't exist"); - accept_spool(); - return; - } - - $incoming->accept($deliver_to); - } - - sub accept_spool { - my $path = "/var/spool/mail/$USER"; - report("defaulting to spool delivery ($path)"); - $incoming->accept($path); - } +my $args_dispatch = merge($defaults, $args); +#print Dumper($args_dispatch); +#exit; -# - - - - - - - - - - - - - - - - - - - - -# helper functions -# - - - - - - - - - - - - - - - - - - - - +my @args_array = %$args_dispatch; +my $dispatcher = Mail::Audit::Dispatch->new(@args_array); +$dispatcher->run(); -sub get_coderef { - my $codepack = shift; - my $method = shift; - $codepack || return '[error]'; - $method ||= ''; - $method && ($codepack .= '::'); - return eval '\&' . $codepack . $method . ';'; -} - -sub export_symbols { - # my $callpack = 'rules'; - # my @EXPORT = qw( incoming subject MAILDIR jaccept ); - # foreach my $sym (@EXPORT) { - no strict 'refs'; - # *{"${callpack}::$sym"} = get_coderef('main', $sym); - # } - { - no strict 'refs'; - *{"rules::jaccept"} = get_coderef('main', 'jaccept'); - *{"rules::report"} = get_coderef('main', 'report'); - *{"rules::compareTarget"} = get_coderef('main', 'compareTarget'); - } - $rules::MAILDIR = $MAILDIR; - $rules::incoming = $incoming; -} - -sub gchomp { - my $str = shift; - chomp($str); - return $str; -} +1;