There is considerable collection of various utilities in the ZMailer sources. Not all of them even become installed into your system in all situations.
vacation automatically replies to incoming mail. The canned reply is contained in the file ~/.vacation.msg, that you should create in your home directory (or the file Msgfile specified by the -m option).
This file should include a header with at least a "Subject:" line (it should not include a "To:" line — if you want, you may include "From:" line, especially if you use the -m option), for example.
To start vacation, run the command vacation start. It will create a ~/.vacation.msg file (if you don't already have one) in your home directory containing the message you want to send people who send you mail, and a ~/.forward file in your home directory containing a line of the form:
"\name", "|/opt/mail/bin/vacation name" |
chmod og-w $HOME $HOME/.forward |
To stop vacation, run the command vacation stop. It will move the ~/.forward file to ~/.vacforward, and the automatic replies will stop.
vacation start vacation stop vacation -I vacation [ -tN ] [ -mMsgfile ] [ -d ] [user] |
\item[{\tt -I}, {\tt -i}] \mbox{} initialize the {\tt .vacation.pag} and {\tt .vacation.dir} files and start vacation. If the {\tt -I} (or {\tt -i}) flag is not specified, vacation tries to reply to the sender. \item[{\tt -tN}] \mbox{} Change the interval between repeat replies to the same sender. The default is one week. A trailing {\tt s}, {\tt m}, {\tt h}, {\tt d}, or {\tt w} scales N to seconds, minutes, hours, days, or weeks respectively. \item[{\tt -mMsgfile}] \mbox{} specifies the file in which the message to be sent is kept. The default is {\tt \$HOME/.vacation.msg}. \item[{\tt -r}] \mbox{} interval defines interval in days when not to answer again to the same sender. (Default is 1 day.) \item[{\tt -d}] \mbox{} disables the list of senders kept in the {\tt .vacation.pag} and {\tt .vacation.dir} files. \end{description} |
Subject: I am on vacation I am on vacation until July 22. If you have something urgent, please contact Joe Jones (joe@blah.utoronto.ca). --john |
No message is sent if the ``user'' specified in the vacation command (if nothing is specified, it uses your username) does not appear explicitly in the ``To:'' or ``Cc:'' lines of the message, which prevents messages from being sent back to mailing lists and causing loops.
A list of senders is kept in the files ~/.vacation.pag and ~/.vacation.dir in your home directory. These are dbm database files. (Note: not all database systems have two files, either may be missing.) The vacation message is in ~/.vacation.msg and the automatic reply is activated by the ~/.forward (and saved in ~/.vacforward) The default vacation message is stored in /opt/mail/vacation.msg
On machines running ZMailer, the ``name'' argument to vacation is optional, and the $USER environment variable is used to determine where to look for the message and the list of previous recipients.
The $SENDER} variable is checked first to determine the reply destination. It is normally set to the SMTP ``MAIL FROM:'' address or equivalent. This is an additional safeguard against sending replies to mailing lists, the PostMaster or the mailer daemon, since standards and common sense dictate that it never points back to an address that could cause a loop. The ``From '' line is used only as a last resort.
The way the ZMailer uses DBM entries is by using strings with their terminating NULL as keys, and as data.. Thus the length is {\tt strlen(string)+1}, not {\tt strlen(string)} !
WARNING: Policy data parsing does use unchecked buffers! \begin{verbatim} Usage: makedb [-a|-p] dbtype database.name [infilename|-] \end{verbatim} Dbtypes are: {\tt ndbm gdbm btree bhash} If no {\tt infilename} is defined, {\tt database.name} is assumed. \begin{description} \item[{\tt NDBM}] \mbox{} appends {\tt .pag}, and {\tt .dir} into the actual db file names. \item[{\tt GDBM}] \mbox{} {\bf does not} append {\tt .gdbm} into the actual db file name. \item[{\tt BTREE}] \mbox{} {\bf does not} append {\tt .db} into the actual db file name. \item[{\tt BHASH}] \mbox{} appends {\tt .pag}, and {\tt .dir} into the actual db file names. \end{description} The {\tt -a} option is for parsing input that comes in {\tt aliases} format: {\tt key: data,in,single,long,line} |
The way the ZMailer uses DBM entries is by using strings with their terminating {\tt NULL} as keys, and as data.. Thus the length is {\tt strlen(string)+1}, not {\tt strlen(string)} ! \begin{verbatim} Usage: dblook [-dump] dbtype database.name [key] \end{verbatim} Dbtypes are: {\tt ndbm gdbm btree bhash} \begin{description} \item[{\tt NDBM}] \mbox{} appends {\tt .pag}, and {\tt .dir} into the actual db file names. \item[{\tt GDBM}] \mbox{} {\bf does not} append {\tt .gdbm} into the actual db file name. \item[{\tt BTREE}] \mbox{} {\bf does not} append {\tt .db} into the actual db file name. \item[{\tt BHASH}] \mbox{} appends {\tt .pag}, and {\tt .dir} into the actual db file names. \end{description} %\end{multicols} % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% \clearpage \subsection{{\tt policy-builder.sh}} \begin{alltt} #! /bin/sh # # Sample smtp-policy-db builder script. # # This merges following files from $MAILVAR/db/ directory: # smtp-policy.src # localnames ('= _localnames') # smtp-policy.relay ('= _full_rights') # smtp-policy.mx ('relaytargets +') # smtp-policy.spam ('= _bulk_mail') # smtp-policy.spam.manual ('= _bulk_mail') # # These all together are used to produce files: smtp-policy.$DBEXT # The produced database retains the first instance of any given key. # #FLAG= #while getopts n c; do # case $c in # n) FLAG=$c;; # ?) exit 2;; # esac #done #shift `expr $OPTIND - 1` if [ "x$1" = "x-n" ]; then FLAG=n shift fi if [ "x$1" = "x?" ]; then exit 2 fi ZCONFIG=@ZMAILERCFGFILE@ . $ZCONFIG umask 022 cd $MAILVAR/db if [ ! -f smtp-policy.src ] ; then echo "No $MAILVAR/db/smtp-policy.src input file" exit 64 # EX_USAGE fi if [ f$FLAG != fn ]; then if [ -x $MAILBIN/smtp-policy-retrieve.pl ] ; then $MAILBIN/smtp-policy-retrieve.pl else if [ -x $MAILBIN/spamlist.py ] ; then $MAILBIN/spamlist.py > smtp-policy.spam.new && \verb/\/ mv smtp-policy.spam.new smtp-policy.spam else # # Following IS NOT SAFE, if either produces errors, those # go (usually) to the result file, and in the end the result # OVERWRITES the "running" smtp-policy.spam file. # > smtp-policy.spam.new lynx -source http://www.webeasy.com:8080/spam/spam_download_table \verb/\/ >> smtp-policy.spam.new lynx -source http://www.sprocket.com/Security/SpamDomains | \verb/\/ awk '\{print $1\}' >> smtp-policy.spam.new cat smtp-policy.spam.new | sed 's/^@//g' | tr "[A-Z]" "[a-z]" | \verb/\/ sort | uniq > smtp-policy.spam.new2 if [ `grep -c cyberpromo smtp-policy.spam.new` -gt "0" ]; then mv smtp-policy.spam smtp-policy.spam.old mv smtp-policy.spam.new2 smtp-policy.spam rm -f smtp-policy.spam.new else echo "Hmm....something went wrong while updating the spam policy." echo "Please try again." exit 1 fi fi fi fi # Fork off a subshell to do it all... ( # The basic boilerplate cat smtp-policy.src # Localnames cat localnames | \verb/\/ awk '/^#/\{next;\} NF >= 1 \{printf "%s = _localnames\verb/\/n",$1;\}' # smtp-policy.relay # (Lists domains and networks that are allowed to use us as relay) if [ -f smtp-policy.relay ] ; then cat smtp-policy.relay | \verb/\/ awk '/^#/\{next;\} \{printf "%s = _full_rights %s %s %s %s %s %s %s %s\verb/\/n", $1,$2,$3,$4,$5,$6,$7,$8,$9;next;\}' fi # smtp-policy.mx # (Lists domains that are allowed to use us as inbound MX relay for them) if [ -f smtp-policy.mx ] ; then cat smtp-policy.mx | \verb/\/ awk '/^#/\{next;\} NF >= 1 \{printf "%s relaytarget +\verb/\/n",$1;\}' fi # smtp-policy.spam # (Lists users, and domains that are known spam sources) # (We use file from "http://www.webeasy.com:8080/spam/spam_download_table" # which is intended for QMAIL, and thus needs to be edited..) if [ -f smtp-policy.spam -o -f smtp-policy.spam.manual ] ; then ( if [ -f smtp-policy.spam ] ; then cat smtp-policy.spam fi if [ -f smtp-policy.spam.manual ] ; then cat smtp-policy.spam.manual fi ) | tr "[A-Z]" "[a-z]" | sed 's/^@//g' | sort | uniq | \verb/\/ awk '/^\verb/\/[/\{ # an address block to reject printf "%s rejectnet +\verb/\/n", $1; next; \} \{ # All other cases are usernames with their domains printf "%s = _bulk_mail\verb/\/n", $1; \}' fi # --------- end of subshell ) > smtp-policy.dat umask 022 # Make sure the resulting db file(s) are readable by all $MAILBIN/makedb -p $DBTYPE smtp-policy-new smtp-policy.dat || exit $? case $DBTYPE in dbm) mv smtp-policy-new.dir smtp-policy.dir mv smtp-policy-new.pag smtp-policy.pag ;; ndbm) mv smtp-policy-new.dir smtp-policy.dir mv smtp-policy-new.pag smtp-policy.pag ;; gdbm) mv smtp-policy-new.gdbm smtp-policy.gdbm ;; btree) mv smtp-policy-new.db smtp-policy.db ;; esac exit 0 \end{alltt} |
The autoanswer program is intended to be placed into system global aliases database as following entry:
autoanswer: "| /path/to/MAILBIN/autoanswer" |
It yields a reply message for all, except the error messages, nor to those with X-autoanswer-loop: header in them.
The reply sends back the original incoming message headers in the message body along with some commentary texts.
The program is, in reality, a perl script which can easily be tuned to local needs.
#!@PERL@ ########################################################################## # # Autoanswer.pl 1.0 for ZMailer 2.99.48+ # (C) 1997 Telecom Finland # Valtteri Karu <valtteri.karu@tele.fi> # # This program sends autoreply and the original headers to the originator # of the message. Version 2.99.48+ of the Zmailer is required for detecting # possible false addresses. # # USAGE: # # Create an alias for the address to use: # autoreply: "|/path/to/autoanswer.pl" # ########################################################################## $nosend = 0; $double = 0; $address = $ENV{'SENDER'}; if( ! -r "$ENV{'ZCONFIG'}") { LOG("zmailer.conf missing"); exit 2; } open(ZMAILER,"< $ENV{'ZCONFIG'}" ); while(<ZMAILER>) { chomp; split(/=/); $ZMAILER{$_[0]}=$_[1]; } close ZMAILER; $logfile = $ZMAILER{'LOGDIR'} . "/autoanswer"; while (<STDIN>) { $text = $_; if (($text eq "\n") && ( $double = 1)) { last; } if (($text eq "\n") && ( $double = 0)) { $double = 1; next; } if ($text =~ "X-autoanswer-loop: ") { $nosend = 1; LOG("Looping message, sender=$address"); } $double = 0; push(@header,$text); } if (($address eq '<>') || ($nosend = 0)) { LOG("SENDER invalid"); exit 1; } $outfile = $ZMAILER{'POSTOFFICE'} . "/public/autoanswer.$$"; #$outfile = "/tmp/aa.$$"; $now = time; $txttime = localtime(time); open(OUT,">$outfile"); select(OUT); print "channel error\n"; print "to $address\n"; print "env-end\n"; print "From: Autoreply service <postmaster>\n"; print "To: $address\n"; print "Subject: Autoreply\n"; print "X-autoanswer-loop: Megaloop \n\n"; print " This is autoreply answer message by your request.\n\n"; print " Original message was received at UNIX time $now;\n"; print " which means '$txttime' in cleartext.\n\n"; print " Headers were:\n\n"; print "--------------------------------------------------------------------\n"; print @header; print "--------------------------------------------------------------------\n"; print "\n Have a nice day.\n"; select(STDOUT); close OUT; $inode=(stat($outfile))[1]; $newfile=$ZMAILER{'POSTOFFICE'} . "/router/$inode"; rename($outfile, $newfile); LOG("Sent to $address"); exit 0; sub LOG { open(LOGf, ">>$logfile"); $ttime = localtime(time); printf (LOGf "$ttime autoanswer: @_\n"); close LOGf; } |
This is elementary wrapper script building binary databases with makedb utility into a temporary file, and replacing the old files with the new ones in proper order for the router's automatic source change detecting relation parameter {\tt -m} to work correctly.
{\bf Usage} \begin{verbatim} newdb /db/path/name [-u|-l] [input-file-name] \end{verbatim} This script uses system {\tt ZCONFIG} file to find out the desired database type, and derives the actual database file names from the variable. Suffix selection rules are: \begin{verbatim} dbm .pag and .dir ndbm .pag and .dir gdbm .gdbm btree .db \end{verbatim} |