libtool/config/mailnotify
Scott James Remnant 0964364db2 * ltmain.in, libtoolize.in, commit, config/mailnotify: Correctly
quote usage of $basename and $dirname to avoid (unlikely) path
expansion.
2004-02-13 07:26:08 +00:00

442 lines
12 KiB
Bash
Executable File

#!/bin/sh
#
# mailnotify (GNU cvs-utils) version 0.2
# Written by Gary V. Vaughan <gary@gnu.org>
# Copyright (C) 2004 Free Software Foundation, Inc.
# This is free software; see the source for copying conditions. There is NO
# warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful, but
# WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
# General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, a copy can be downloaded from
# http://www.gnu.org/copyleft/gpl.html, or by writing to the Free
# Software Foundation, Inc., 59 Temple Place - Suite 330, Boston,
# MA 02111-1307, USA.
# Usage: $progname [OPTION]... [--] to-address...
#
# -C ADDR --carbon-copy=ADDR send a carbon-copy to ADDR
# -F ADDR --from=ADDR override default from address with ADDR
# -f FILE --filename=FILE content of this part
# -m TYPE --mime-type=TYPE mime-type of this part
# -n another mime part (-f, -m) to follow
# -o FILE --output-file=FILE output to FILE instead of sending
# -s TEXT --subject=TEXT set subject header
# -v --verbose run in verbose mode
# --version print version information
# -h,-? --help print short or long help message
# Assemble a (possibly multi-part) mime message and hand it to the local
# sendmail for onward delivery. MUAs tend to mangle patch attachments in
# various ways: not setting the mime-type correctly, line wrapping the
# patch itself, escaping certain values, etc. This script is designed to
# make it easier to send a patch as a MIME attachment, though it is general
# enough that it might be useful otherwise.
# For example to send a patch as an attachment, assuming the patch itself
# is in PATCHFILE:
#
# echo 'Applied to HEAD' > body
# $progname -f body -m text/plain -n -f PATCHFILE -m text/x-patch \
# -s 'FYI: PATCHFILE' patch-list@foo.org
# You will probably find using this script in conjunction with clcommit
# or cvsapply will save you an awful lot of typing.
# Report bugs to <gary@gnu.org>
: ${TMPDIR=/tmp}
: ${HOST=`hostname`}
: ${MV="mv -f"}
: ${RM="rm -f"}
: ${SED="sed"}
dirname="s,/[^/]*$,,"
basename="s,^.*/,,g"
# Work around backward compatibility issue on IRIX 6.5. On IRIX 6.4+, sh
# is ksh but when the shell is invoked as "sh" and the current value of
# the _XPG environment variable is not equal to 1 (one), the special
# positional parameter $0, within a function call, is the name of the
# function.
progpath="$0"
# The name of this program:
progname=`echo "$progpath" | $SED "$basename"`
PROGRAM=mailnotify
# Global variables:
EXIT_SUCCESS=0
EXIT_FAILURE=1
multipart=1
outputfile=""
exit_cmd=:
sed_mail_address='s,^.*<\(.*\)>.*$,\1,'
# func_echo arg...
# Echo program name prefixed message.
func_echo ()
{
echo $progname: ${1+"$@"}
}
# func_error arg...
# Echo program name prefixed message to standard error.
func_error ()
{
echo $progname: ${1+"$@"} 1>&2
}
# func_fatal_error arg...
# Echo program name prefixed message to standard error, and exit.
func_fatal_error ()
{
func_error ${1+"$@"}
exit $EXIT_FAILURE
}
# func_verbose arg...
# Echo program name prefixed message in verbose mode only.
func_verbose ()
{
$opt_verbose && func_error ${1+"$@"}
}
# func_fatal_help arg...
# Echo program name prefixed message to standard error, followed by
# a help hint, and exit.
func_fatal_help ()
{
func_error ${1+"$@"}
func_fatal_error "Try \`$progname --help' for more information."
}
# func_missing_arg argname
# Echo program name prefixed message to standard error and set global
# exit_cmd.
func_missing_arg ()
{
func_error "missing argument for $1"
exit_cmd=exit
}
# Echo short help message to standard output and exit.
func_usage ()
{
$SED '/^# Usage:/,/# -h/ {
s/^# //; s/^# *$//;
s/\$progname/'$progname'/;
p;
}; d' < "$progpath"
echo
echo "run \`$progname --help | more' for full usage"
exit $EXIT_SUCCESS
}
# func_help
# Echo long help message to standard output and exit.
func_help ()
{
$SED '/^# Usage:/,/# Report bugs to/ {
s/^# //; s/^# *$//;
s/\$progname/'$progname'/;
p;
}; d' < "$progpath"
exit $EXIT_SUCCESS
}
# func_version
# Echo version message to standard output and exit.
func_version ()
{
$SED '/^# '$PROGRAM' (GNU /,/# warranty; / {
s/^# //; s/^# *$//;
s/\((C)\)[ 0-9,-]*\( [1-9][0-9]*\)/\1\2/;
p;
}; d' < "$progpath"
exit $EXIT_SUCCESS
}
# Parse options once, thoroughly. This comes as soon as possible in
# the script to make things like `mailnotify --version' happen quickly.
{
# sed scripts:
my_sed_single_opt='1s/^\(..\).*$/\1/;q'
my_sed_single_rest='1s/^..\(.*\)$/\1/;q'
my_sed_long_opt='1s/^\(--[^=]*\)=.*/\1/;q'
my_sed_long_arg='1s/^--[^=]*=//'
while test $# -gt 0; do
opt="$1"
shift
case $opt in
-C|--carbon-copy) test $# -eq 0 && func_missing_arg "$opt" && break
cc="$1"
shift
;;
-F|--from) test $# -eq 0 && func_missing_arg "$opt" && break
from="$1"
shift
;;
-f|--filename) test $# -eq 0 && func_missing_arg "$opt" && break
if test -f "$1"; then :; else
func_error "\`$1' does not exist"
exit_cmd=exit
break
fi
eval datafile$multipart=\"$1\"
shift
;;
-m|--mime-type) test $# -eq 0 && func_missing_arg "$opt" && break
case $1 in
text/*) ;;
*/*) func_error "only text/* mime-types supported"
;;
*) func_error "invalid mime-type, \`$1'"
exit_cmd=exit
;;
esac
eval ctype$multipart=\"$1\"
shift
;;
-n) if eval test -z \"\$ctype$multipart\" ||
eval test -z \"\$datafile$multipart\"; then
func_fatal_error "One part is incomplete -- each part needs a filename and a mime-type"
fi
multipart=`expr 1 + $multipart`
;;
-o|--output-file) test $# -eq 0 && func_missing_arg "$opt" && break
outputfile="$1"
shift
;;
-s|--subject) test $# -eq 0 && func_missing_arg "$opt" && break
subject="$1"
shift
;;
-v|--verbose) opt_verbose=: ;;
# Separate optargs to long options:
--carbon-copy=*|--from=*|--filename=*|--mime-type=*|--output-file=*|--subject=*)
arg=`echo "$opt" | $SED "$my_sed_long_arg"`
opt=`echo "$opt" | $SED "$my_sed_long_opt"`
set -- "$opt" "$arg" ${1+"$@"}
;;
# Separate optargs to short options:
-C*|-F*|-f*|-m*|-o*|-s*)
arg=`echo "$opt" |$SED "$my_sed_single_rest"`
opt=`echo "$opt" |$SED "$my_sed_single_opt"`
set -- "$opt" "$arg" ${1+"$@"}
;;
# Separate non-argument short options:
-n*|-v*)
rest=`echo "$opt" |$SED "$my_sed_single_rest"`
opt=`echo "$opt" |$SED "$my_sed_single_opt"`
set -- "$opt" "-$rest" ${1+"$@"}
;;
-\?|-h) func_usage ;;
--help) func_help ;;
--version) func_version ;;
--) break ;;
-*) func_fatal_help "unrecognized option \`$opt'" ;;
*) set -- "$opt" ${1+"$@"}; break ;;
esac
done
test $# -gt 0 ||
func_fatal_help "no destination address"
if test -z "$outputfile"; then
if test -z "$subject" ||
eval test -z \"\$ctype$multipart\" ||
eval test -z \"\$datafile$multipart\"; then
func_fatal_error "if output is not directed to a file -s, -f, and -m are all required"
fi
else
eval test -n \"\$datafile$multipart\" ||
func_fatal_error "-f is required."
eval test -n \"\$ctype$multipart\" ||
func_fatal_error "with output directed to a file, -m is required"
fi
eval test -f \"\$datafile$multipart\" ||
eval func_fatal_error \"\$datafile$multipart: file not found\"
}
# func_headers outfile destination
# Generate common headers to OUTFILE, where DESTINATION is a comma
# separated list of fully qualified destination addresses.
func_headers ()
{
my_outfile="$1"
my_destination="$2"
my_sed_version_no='/^# '$PROGRAM' (GNU / { s/^# .*version //; p; }; d'
{
echo "User-Agent: $PROGRAM/`$SED \"$my_sed_version_no\" < $progpath`"
echo "MIME-Version: 1.0"
test -n "$from" && echo "From: $from"
echo "To: $my_destination"
test -n "$cc" && echo "CC: $cc"
test -n "$subject" && echo "Subject: $subject"
} > "$my_outfile"
}
# func_single_content outfile
# Send the only message part as a single mime mail part.
func_single_content ()
{
my_outfile="$1"
cat >> "$my_outfile" <<EOF
Content-Type: $ctype1
Content-Transfer-Encoding: 7bit
`cat $datafile1`
EOF
}
# func_multipart_content outfile
# Send the various message parts to OUTFILE as a multipart mime mail.
func_multipart_content ()
{
my_outfile="$1"
boundary="Boundary-${HOST}-$$-`date | tr ' :' -`"
cat <<EOF >> "$my_outfile"
Content-Type: Multipart/Mixed;
boundary="$boundary"
This is a multimedia message in MIME format. If you are reading
this prefix, your mail reader does not understand MIME. You may
wish to look into upgrading to a mime-aware mail reader.
EOF
i=0
while test $i -lt $multipart
do
i=`expr 1 + $i`
{
echo ""
echo "--$boundary"
eval echo \"Content-Type: \$ctype$i\"
echo "Content-Transfer-Encoding: 7bit"
echo ""
eval cat \"\$datafile$i\"
} >> "$my_outfile"
done
{
echo ""
echo "--${boundary}--"
echo ""
} >> "$my_outfile"
}
# func_sendmail infile destination [from]
# Send the message in INFILE to the space delimited list of destination
# addresses in DESTINATION. If FROM is given, it is the address the
# mail purports to be from.
# BEWARE: Many MTAs will refuse mail where FROM does not match the actual
# sending domain.
func_sendmail ()
{
my_infile="$1"
my_destination="$2"
my_from="$3"
from_name=`echo "$my_from" | sed 's, *<.*> *$,,;s,",,g'`
from_addr=`echo "$my_from" | sed "$sed_mail_address"`
SENDMAIL=sendmail
for try_sendmail in sendmail /usr/lib/sendmail /usr/sbin/sendmail; do
if which $try_sendmail >/dev/null; then
SENDMAIL=$try_sendmail
break
fi
done
func_verbose "Delivering mail, please wait..."
if test -n "$from_name"; then
$SENDMAIL -F "$from_name" -f "$from_addr" $my_destination < "$my_infile"
elif test -n "$from_addr"; then
$SENDMAIL -f "$from_addr" $my_destination < "$my_infile"
else
$SENDMAIL $my_destination < "$my_infile"
fi
if test $? -eq 0; then
func_verbose "...succeeded."
rm $my_infile
else
func_fatal_error "Mail delivery failed, draft mail is in $my_infile"
fi
}
## ----- ##
## main. ##
## ----- ##
{
fname="${TMPDIR}/$PROGRAM$RANDOM-$RANDOM.$$"
trap 'rm -f "$fname"; exit 1' 1 2 15
destination=""
for to;
do
case $destination in
"") destination="$to" ;;
*) destination="$destination, $to" ;;
esac
done
func_headers "$fname" "$destination"
if test $multipart -gt 1; then
func_multipart_content "$fname"
else
func_single_content "$fname"
fi
if test -z "$outputfile"; then
destination=""
for to; do
to_addr=`echo "$to" | sed "$sed_mail_address"`
test -n "$to_addr" || to_addr="$to"
destination="$destination $to_addr"
done
func_sendmail "$fname" "$destination" "$from"
else
mv $fname $outputfile || exit $EXIT_FAILURE
fi
}
exit $EXIT_SUCCESS
# Local Variables:
# mode:shell-script
# sh-indentation:2
# End: