$Lock_pathname $PeerAddr $Socket_timeout $DB_table $DBI_database $DBI_user $DBI_pwd
$sendmail_orig @addr_addon &FromAddress @h_rcpt @h_from
%audit_profile @sms_squeezes @alternates_host @dnsbl_whitelist
+
+ $Audit $is_pgp $opt_F $procmailFROM_MAILER $store_ignore $store_ignorenewmail
+ $store_profile
);
require Mail::Alias;
-sub headerhas;
-sub store;
-sub headeris;
-sub did;
-sub dnsbl;
-sub store_muttrc_alternates;
+BEGIN {
+ for (qw(headerhas store headeris did dnsbl store_muttrc_alternates Received_for parts_linear mime_type
+ body_first mimehead spamassassin header_remap)) {
+ eval 'sub '.$_.' { return ::'.$_.'(@_); }';
+ }
+ }
# perlmail-accept & perlmail-sendmail
-our $HOME="/home/lace";
+# Various configuration files location is derived from it:
+our $HOME="/home/USERNAME";
# perlmail-accept
+# Mail folder:
our $Mail="$HOME/Mail";
-our @ValidUsers=qw(root lace short kratochvil _local);
+# Users respected for the 'idle' state (see $IdleMax):
+our @ValidUsers=qw(root USERNAME _local);
+# Maximum number of local console idle seconds while still considered as 'active user':
our $IdleMax=10;
-our $MaxBodySMS=0x1000; # max bytes to pass to Lingua::EN::Squeeze
-our @SMSwebRcpt=qw(420 602 431329);
-our $SMSwebRcpt_username="lace2";
+# Maxium number of bytes to pass to Lingua::EN::Squeeze (performance optimization):
+our $MaxBodySMS=0x1000;
+# Telephone number to send SMSes by WWW::SMS to:
+our @SMSwebRcpt=qw(123 456 789123);
+# Some WWW::SMS modules require username:
+our $SMSwebRcpt_username="SMSUSERNAME";
# perlmail-submit
+# Global system lock for exclusive $DB_table access:
our $Lock_pathname="/tmp/PerlMail.lock";
-our $PeerAddr="exuhome.dyn.jankratochvil.net.:852";
+# 'workstation' hostname and port. Hostname may be dyndns:
+our $PeerAddr="workstation.hostname.tld.:852";
#our $PeerAddr="127.0.0.1:2852";
-our $Socket_timeout=7600; # 15sec is NOT enough!
+# 15sec is NOT enough as the remote peer must complete mail store:
+our $Socket_timeout=7600;
+# MySQL table name:
our $DB_table="PerlMail_folder";
-our $DBI_database="short";
-our $DBI_user="short";
+# MySQL database name:
+our $DBI_database="USERNAME";
+# MySQL user name:
+our $DBI_user="USERNAME";
+# MySQL user password:
our $DBI_pwd=$ENV{"HOME"}."/priv/mysql.".$DBI_user.".pwd";
# perlmail-sendmail
+# Pathname of the original sendmail(8) binary:
our $sendmail_orig=(-x ($_="/usr/sbin/sendmail-orig") ? $_ : "/usr/sbin/sendmail");
+# List of addresses to locally Bcc all mails to:
# Mail-Alias-1.12 defaults to "/etc/mail/aliases" which does not exist on RedHat sendmail-8.12.5-7
# Mail-Alias-1.12 will clutter $_ !
our @addr_addon=(Mail::Alias->new("/etc/aliases")->exists("sentout") ? ("sentout") : ());
our $opt_F; # imported
our $is_pgp; # imported
+# Generate new From address for the target $rcpt of type Mail::Address.
+# $iserror is true for "MAIL FROM" RFC821 address, false for "From:" RFC822 address.
+# Returns: Mail::Address instance.
sub FromAddress
{
my($rcpt,$iserror)=@_;
- my $phrase=(defined $opt_F ? $opt_F : "Jan Kratochvil");
- {
- last if !$is_pgp;
- last if $iserror;
- local *F;
- local $_;
- my $filename="$HOME/.gnupg/options";
- open F,$filename or do { warn "Open \"$filename\": $!"; last; };
- local $/="\n";
- my @keys=map((/^\s*default-key\s+(\S+)\s*$/),<F>);
- @keys==1 or do { warn "Found ".scalar(@keys)." 'default-key's in your \"$filename\", ignoring"; last; };
- close F or warn "Close \"$filename\": $!";
- my $default_key=$keys[0];
- $default_key=~/^[[:xdigit:]]{8}$/ or do { warn "Invalid 'default-key', ignoring: $default_key"; last; };
- return Mail::Address->new(
- $phrase,
- 'pgp-'.uc($default_key).'@jankratochvil.net',
- );
- }
- # !$is_pgp or fallback
- return Mail::Address->new(
- $phrase,
- (!$iserror ? 'rcpt' : 'rcpterr')
- .'-'
- .(defined($rcpt->user()) ? $rcpt->user() : "NOUSER")
- .".AT."
- .(defined($rcpt->host()) ? $rcpt->host() : "LOCAL")
- .'@jankratochvil.net',
- );
+ my $phrase=(defined $opt_F ? $opt_F : "FULL NAME");
+ return Mail::Address->new($phrase,'USER@DOMAIN.TLD');
}
# RedHat sendmail-8.9.3-20/src/conf.c/HdrInfo[]/\Q/* destination fields */\E
# My-Audit
+# Setup profile names.
+# First element of /^=/ form copies it referenced profile to be extended.
+# 'did' =>did() subroutine will return true for it.
+# 'syslog' =>Use syslog(3).
+# 'bell' =>Bell sound.
+# 'sms=\d+'=>Send SMS by WWW::SMS with specified maximum # of parts
our %audit_profile=(
"btw" =>[],
"silent"=>["=btw" ,"did"],
"sms" =>["=bell" ,"sms=1"],
"crit" =>["=sms" ,"sms=3"],
);
+# Try the squeezing methods in this order:
our @sms_squeezes=(
{ "SqueezeControl"=>"noconv" },
{ "SqueezeControl"=>"conv" ,"SQZ_OPTIMIZE_LEVEL"=>0 },
{ "SqueezeControl"=>"max" ,"SQZ_OPTIMIZE_LEVEL"=>0 },
{ "SqueezeControl"=>"max" ,"SQZ_OPTIMIZE_LEVEL"=>1 },
);
+# Hostnames where we had old/alternate e-mail addresses:
our @alternates_host=(
- "jabberwock.ucw.cz", # short@ucw.cz
- "atrey.karlin.mff.cuni.cz", # short@atrey.karlin.mff.cuni.cz
- "k332.feld.cvut.cz", # short@k332.feld.cvut.cz
);
+# Override DNS blacklists:
our @dnsbl_whitelist=(
- "195.250.128.83", # smtp3.vol.cz; vol.cz.multistage.blackholes.five-ten-sg.com.
"64.49.222.22", # mail.pm.org: rackspace.com.spam-support.blackholes.five-ten-sg.com.
"208.147.243.5", # gambit.liquidcomm.net: cw.net.spam-support.blackholes.five-ten-sg.com.
- "213.235.135.70", # smtp.tiscali.cz: tiscali.cz.multistage.blackholes.five-ten-sg.com.
"205.139.198.11", # eniac.disaster.com: cw.net.spam-support.blackholes.five-ten-sg.com.
"127.0.0.2", # 2.0.0.127.relays.ordb.org.
"65.113.40.131", # bozo.vmware.com: qwest.net.spam-support.blackholes.five-ten-sg.com.
"66.218.85.33", # mta2.wss.scd.yahoo.com: yahoo.com.spam.blackholes.five-ten-sg.com.
- "212.80.76.42", # mx2.seznam.cz: seznam.cz.free.blackholes.five-ten-sg.com.
);
our $Audit; # imported
our $store_ignore; # imported
sub audit
{
- # TODO: <short-m@> storage?
+ $store_profile=undef();
# never spawn new mail if FROM_MAILER
# $isFROM_MAILER postponed after maillists as they may look as FROM_MAILER
# spam honeypots
return if did sub {
local $_;
- local $store_profile="silent";
- store "=spam" if grep /^\Qshort\@k332.feld.cvut.cz\E/i,Received_for();
- # TODO: foreign violation of RFC 822 section 4.4.4, Subject:.*Automatick.+odpov.+v.+nep.+tomnosti
+ # Do not local $store_file as it is our-imported
+ $store_profile="silent";
store "=spam" if headeris "From",'<ghandchi@hotmail.com>';
- store "=spam" if headeris "From",'<newsletter@levnapc.cz>';
- store "=spam" if headeris "From",'<Tomas@dtpstudio.cz>';
- store "=spam" if headeris "From",'<BNcom@email.bn.com>';
- store "=spam" if headeris "From",'<e4luck@lists.opt4email.com>';
store "=spam" if headeris "From",'<mailcontests@lists.servitall.com>';
- store "=spam" if headeris "From",'<canda@lica.cz>';
{
# weak detection: files with text/html w/o text/plain are usually a spam
my @types_linear=map({ mime_type($_); } parts_linear());
}
store "=spam" if ($_=mimehead(body_first())->mime_attr("Content-Type.charset")) && /^big5/i;
};
+ $store_profile=undef();
# spam detection
return if did sub {
- local $store_profile="silent";
+ # Do not local $store_file as it is our-imported
+ $store_profile="silent";
local $_;
- store "=spam".($_ eq 1 ? "" : ";$_") if $_=razor2();
+ store "=spam".($_ eq 1 ? "" : ";$_") if $_=spamassassin();
};
+ $store_profile=undef();
return if did sub {
- local $store_profile="silent";
+ # Do not local $store_file as it is our-imported
+ $store_profile="silent";
local $_;
store "=spam" .";$_","log" if $_=dnsbl '.relays.ordb.org.' ,1; # all hosts
store "=spam" .";$_","log" if $_=dnsbl '.blackholes.mail-abuse.org.' ,1; # all hosts
# I don't send viruses but viruses propagate mails of mine
store "=spam" if headeris "X-Mailer",'ravmd/8.3.2';
};
-
- # special delivery
- store "=err","bell" and return if headerhas \&Received_for,'<short+err@>';
+ $store_profile=undef();
# ppl-wished foreign remapping, Reply-To is left untouched!
# FIXME: modifications are now being dropped by &write_message!
header_remap("From",{
- 'kerere@post.cz' =>'kamzik@k332.feld.cvut.cz',
- 'profes@mbox.vol.cz' =>'kratochvilova@egp.cz',
- 'jkrouzek@mbox.vol.cz' =>'krouzek@mbox.fsv.cuni.cz',
- 'jakub.gorner@lidovky.cz' =>'tonda@disnet.cz',
- 'jan.kolar@videoprogress.cz' =>'jenda.kolar@volny.cz',
- 'daniel.rulicek@cponline.cz' =>'daniel.rulicek@cpress.cz',
- 'pavel@suse.cz' =>'pavel@ucw.cz',
+ 'RECEIVED@ADDRESS.TLD'=>'MAP_TO@ADDRESS.TLD',
});
# My obsolete e-mail addresses
# nasty public lists with $store_ignore
{
- local $store_profile="log";
+ # Do not local $store_file as it is our-imported
+ $store_profile="log";
local $store_ignore;
$store_ignore="smsmail" if 1==$Audit->body() && length(join "",$Audit->body())<180; # SMS mail
- $store_ignore="sms OS" if $Audit->subject()=~/^Email pro: /; # "^Email pro: gsm@sh\.cvut\.cz$";
$store_ignore="list-moron" if grep { headeris "From",$_; } qw(
- <kempny@>
- <help.me@wo.cz>
- <mr.death@mail.cz>
- <danx3@centrum.cz>
- <@mujoskar.cz>
- <satko@quanto.nr.sanet.sk>
- <vithous@attorney.cz> <viroman@attorney.cz>
- <konf@klain.cz>
- <stein@tiscali.cz>
- <barevnej@volny.cz>
+ <SOMEONE@SOMEWHERE.COM>
);
- store "=gsm" if headeris "Sender" ,'<owner-gsm@sh.cvut.cz>';
- store "=gsmpand" if headeris "List-Post",'<gsm@pandora.cz>';
- store "=9kc","log" if headeris "List-Post",'<n9k@pandora.cz>';
- store "=9kcd","log" if headeris "List-Post",'<dev9k@pandora.cz>';
+ store "=LOCALNAME" if headeris "Sender" ,'<OWNER-XYZZY@DOMAIN.TLD>';
}
+ $store_profile=undef();
# lists
- store "=mozillabug","log" if headeris "From" ,'<bugzilla-daemon@mozilla.org>';
- store "=9ku","log" if headeris "List-Id" ,'<9000.listman.net>';
- store "=9kd","log" if headeris "Sender" ,'<owner-9000-developers@geekstuff.co.uk>';
- store "=spong","log" if headeris "List-Id" ,'<spong-users.lists.sourceforge.net>';
- store "=gtkd","silent" if headeris "List-Id" ,'<gtk-devel-list.gnome.org>';
- store "=gnomevfs","log" if headeris "List-Id" ,'<gnome-vfs-list.gnome.org>';
- store "=mffstatnice","bell" if headeris "List-Post",'<statnice@atrey.karlin.mff.cuni.cz>';
- store "=hw","log" if headeris "List-Post",'<hw-news@list.gin.cz>';
- store "=gnokii","log" if headeris "List-Id" ,'<gnokii-users.mail.freesoftware.fsf.org>';
- store "=winelic","log" if headeris "List-Id" ,'<wine-license.winehq.com>';
- store "=wined","silent" if headeris "List-Id" ,'<wine-devel.winehq.org>';
- store "=winepat","silent" if headeris "List-Id" ,'<wine-patches.winehq.com>';
- store "=winecvs","silent" if headeris "List-Id" ,'<wine-cvs.winehq.com>';
- store "=wineann","log" if headeris "List-Id" ,'<wine-announce.winehq.com>';
- store "=ros","log" if headeris "List-Id" ,'<ros-general.reactos.geldorp.nl>';
- store "=roskernel","log" if headeris "List-Post",'<ros-kernel@reactos.com>';
- store "=roscvs","silent" if headeris "List-Post",'<ros-cvs@reactos.com>';
- store "=rosbug","log" if headeris "Reply-To" ,'<scarab@reactos.wox.org>';
- store "=fsd","silent" if headeris "X-Mailing-List",'<linux-fsdevel@vger.kernel.org>';
- store "=kerneld","silent" if headeris "X-Mailing-List",'<linux-kernel@vger.kernel.org>';
- store "=surprise","sms" if headeris "List-Post",'<surprise@atrey.karlin.mff.cuni.cz>';
- store "=surprisesuse","sms" if headeris "Sender" ,'<owner-surprise@suse.cz>';
- store "=tacacs","log" if headeris "Sender" ,'<tacplus-l@disaster.com>';
- store "=tacacs","log" if headeris "Sender" ,'<owner-tacplus-l@disaster.com>';
- store "=tacacs","log" if headeris "List-Id" ,'<devel.lists.tacplus.org>';
- store "=pm","sms" if headeris "Sender" ,'<owner-prague-pm@pm.org>';
- store "=radary","log" if headeris "Reply-To" ,'<pha@radary.cz>';
- store "=dnet","log" if headeris "Sender" ,'<@lists.distributed.net>';
- store "=linux-input","log" if headeris "List-Post",'<linux-input@atrey.karlin.mff.cuni.cz>';
- store "=strom","bell" if headeris "List-Post",'<vodni-strom@atrey.karlin.mff.cuni.cz>';
- store "=netinfo","log" if headeris "Sender" ,'<owner-netinfo-l@vol.cz>';
- store "=saintmj","log" if headeris "From" ,'<netsaint@kam-enterprise.ms.mff.cuni.cz>';
- store "=saintmj","log" if headeris "From" ,'<netsaint@kam.mff.cuni.cz>';
- store "=4cerr","bell" if headeris "From" ,'<owner-4cinfo@atrey.karlin.mff.cuni.cz>';
- store "=4c","sms" if headeris "List-Post",'<4cinfo@atrey.karlin.mff.cuni.cz>';
- store "=slashdot","bell" if headeris "From" ,'<slashdot@slashdot.org>';
- store "=freshmeat","bell" if headeris "From" ,'<noreply@freshmeat.net>';
- store "=sourceforge","bell" if headeris "From" ,'<noreply@sourceforge.net>';
- store "=gsmperlcvs","silent" if headeris("From" ,'<johan@intra.tektonica.com>')
- && $Audit->subject()=~/^'.*' has been updated!$/;
- store "=libtoold","log" if headeris "List-Id" ,'<libtool.gnu.org>';
- store "=libtoolpat","log" if headeris "List-Id" ,'<libtool-patches.gnu.org>';
- # own webs
- store "=energie","bell" if headeris "From" ,qr/^EnergieWeb/;
- store "=ats","log" if headeris("From" ,'<root@ms.atspraha.cz>')
- || (headeris("From",'<online@ringier.cz>') && headerhas("To",'<blesk@atspraha.cz>'));
- store "=atscasablanca","log" if headeris "From" ,'<casablanca@ms.atspraha.cz>';
- store "=www-sms","log" if headeris "List-Id" ,'<www-sms-developers.lists.sourceforge.net>';
-
- # Petr Koutecky does not mark his Stuff
- store "=koutecky","log" if headeris "From" ,'<velkyhroch@seznam.cz>';
+ store "=LIST","log" if headeris "List-Id" ,'<SOME.ID.OF.THE.LIST>';
store "=errm","bell" if $isFROM_MAILER && !did();
return $_;
}
+# $args{"from"}
+# $args{"subject"}
+# $args{"body"}
sub audit_sms
{
my(%args)=@_;