v0.9.3
This commit is contained in:
parent
245a01d527
commit
6b2298628a
@ -1,6 +1,14 @@
|
||||
Change Log
|
||||
==========
|
||||
|
||||
0.9.2 -> 0.9.3
|
||||
--------------
|
||||
* Proxy mode (--immediate and --as-proxy)
|
||||
* Message preprocessing (--filter)
|
||||
* Message store classes better separated using abstract interfaces
|
||||
* Improved notification script, with MIME encoding
|
||||
* Builds with old 2.91 version of gcc
|
||||
|
||||
0.9.1 -> 0.9.2
|
||||
--------------
|
||||
* Better autoconf detection. (Now builds on all SourceForge's
|
||||
|
3
INSTALL
3
INSTALL
@ -5,7 +5,8 @@ Introduction
|
||||
standard GNU "./configure; make; make install" installation from source.
|
||||
The E-MailRelay userguide describes what you have to do after the "make
|
||||
install" (under GNU/Linux) in order to get the emailrelay daemon to start
|
||||
up at boot-time and automatically forward e-mail to your ISP.
|
||||
up at boot-time, automatically forward e-mail to your ISP when connected,
|
||||
and bounce failed mail.
|
||||
|
||||
Basic Installation
|
||||
==================
|
||||
|
@ -1,2 +1,2 @@
|
||||
EXTRA_DIST =
|
||||
EXTRA_DIST = emailrelay.spec
|
||||
SUBDIRS = src bin lib doc
|
||||
|
@ -69,7 +69,7 @@ PACKAGE = @PACKAGE@
|
||||
RANLIB = @RANLIB@
|
||||
VERSION = @VERSION@
|
||||
|
||||
EXTRA_DIST =
|
||||
EXTRA_DIST = emailrelay.spec
|
||||
SUBDIRS = src bin lib doc
|
||||
ACLOCAL_M4 = $(top_srcdir)/aclocal.m4
|
||||
mkinstalldirs = $(SHELL) $(top_srcdir)/mkinstalldirs
|
||||
@ -276,6 +276,7 @@ distdir: $(DISTFILES)
|
||||
|| exit 1; \
|
||||
fi; \
|
||||
done
|
||||
|
||||
info-am:
|
||||
info: info-recursive
|
||||
dvi-am:
|
||||
|
10
NEWS
10
NEWS
@ -1,10 +1,2 @@
|
||||
|
||||
To do
|
||||
=====
|
||||
|
||||
from 0.9.1
|
||||
----------
|
||||
Better use of autoconf
|
||||
Port to BSD & solaris
|
||||
test IPv6
|
||||
no news
|
||||
|
||||
|
55
README
55
README
@ -1,27 +1,37 @@
|
||||
E-MailRelay
|
||||
===========
|
||||
E-MailRelay Readme
|
||||
==================
|
||||
|
||||
Abstract
|
||||
--------
|
||||
E-MailRelay is a simple store-and-forward SMTP MTA, designed for standalone
|
||||
machines with an intermittent (dial-up) connection to the wider Internet.
|
||||
In most situations the only configuration required is to specify the mail
|
||||
gateway address on the command line.
|
||||
E-MailRelay is a simple SMTP store-and-forward message transfer agent (MTA).
|
||||
It runs as an SMTP server, storing incoming e-mail in a local spool directory,
|
||||
and then forwarding the stored messages to a downstream SMTP server on request.
|
||||
It can also run as a proxy server, forwarding (and optionally pre-processing)
|
||||
incoming e-mail as soon as it is received. It does not do any message routing,
|
||||
other than to a local postmaster. Because of this functional simplicity it is
|
||||
extremely easy to configure, typically only requiring the address of the
|
||||
downstream SMTP server to be put on the command line.
|
||||
|
||||
C++ source code is available for Linux and Windows. Distribution is under
|
||||
the GNU General Public License.
|
||||
C++ source code is available for Linux, FreeBSD and Windows. Distribution is
|
||||
under the GNU General Public License.
|
||||
|
||||
Quick start
|
||||
-----------
|
||||
The "emailrelay" program can be run as an SMTP server daemon using the
|
||||
command "emailrelay --as-server", and stored mail can be forwarded by
|
||||
running the command "emailrelay --as-client <isp-mail-server>:smtp".
|
||||
E-MailRelay can be built and installed from source using the ususal
|
||||
"./configure ; make ; make install" incantation. The program runs as an SMTP
|
||||
server daemon using the "emailrelay --as-server" command, and stored mail is
|
||||
forwarded to a downstream server by running "emailrelay --as-client <isp-mail-server>:smtp".
|
||||
|
||||
The "--as-server" command is typically run automatically at boot time,
|
||||
using the boot scripts under "/etc/init.d" or "/sbin/init.d", while the
|
||||
"--as-client" command is normally put into pppd's "ip-up" script ("/etc/ppp/ip-up")
|
||||
The "--as-server" command is typically run automatically at boot time, using
|
||||
the boot scripts under "/etc/init.d" or "/sbin/init.d", while the "--as-client"
|
||||
command is normally put into pppd's "ip-up" script ("/etc/ppp/ip-up")
|
||||
in place of "sendmail -q".
|
||||
|
||||
The program can also run as a proxy server ("--as-proxy") so that e-mail
|
||||
messages are forwarded immediately to a downstream server -- usually the local
|
||||
system's default MTA. This can be usefully combined with a mail pre-processor
|
||||
program by using the "--filter" switch.
|
||||
|
||||
The program requires a writeable spool directory to store e-mail messages.
|
||||
The directory defaults to "/usr/local/var/spool/emailrelay", but it
|
||||
can be changed by using the "--spool-dir" switch.
|
||||
@ -31,8 +41,8 @@ server is running then it will fail to start up, with a "cannot bind" error
|
||||
message. The port number can be changed by using the "--port" switch.
|
||||
|
||||
To test the program out without a full installation:
|
||||
(1) run the server as "emailrelay --no-daemon --no-syslog --log --spool-dir ${HOME} --port 10001 &"
|
||||
(2) reconfigure your e-mail client to use port 10001 rather than 25 ("smtp")
|
||||
(1) run the server as "emailrelay --no-daemon --no-syslog --log --spool-dir ${HOME} --port 10025 &"
|
||||
(2) reconfigure your e-mail client to use port 10025 rather than 25 ("smtp")
|
||||
(3) send some test messages to addressees on the Internet
|
||||
(4) connect to the Internet
|
||||
(5) forward the stored messages using "emailrelay --spool-dir ${HOME} --as-client <your-isp-smtp-host>:smtp"
|
||||
@ -47,11 +57,13 @@ The following documentation is provided:
|
||||
* doc/userguide.txt -- user guide
|
||||
* doc/reference.txt -- reference document
|
||||
* doc/developer.txt -- developer guide
|
||||
* ChangeLog -- change log for releases
|
||||
|
||||
And for completeness the following stub documents are also included:
|
||||
* NEWS
|
||||
* AUTHORS
|
||||
* ChangeLog
|
||||
|
||||
Source code documentation can be generated by using doxygen (www.doxygen.org).
|
||||
|
||||
Configurations
|
||||
--------------
|
||||
@ -65,3 +77,12 @@ The code was developed on SuSE Linux 7.1 using:
|
||||
and ported to Windows 98 using
|
||||
* MSVC 6.0
|
||||
|
||||
The code has also been built successfully on:
|
||||
* Linux on Alpha hardware (Debian 2.2)
|
||||
* Linux on Sparc hardware
|
||||
* Linux on RS6000 PPC hardware
|
||||
* FreeBSD on Intel hardware
|
||||
* Solaris, using gcc, on Sparc hardware
|
||||
|
||||
(courtesy of SourceForge's compile farm).
|
||||
|
||||
|
@ -1,7 +1,7 @@
|
||||
noinst_SCRIPTS = emailrelay-filter.sh emailrelay-test.sh
|
||||
noinst_SCRIPTS = emailrelay-doxygen-filter.sh emailrelay-test.sh
|
||||
libexec_SCRIPTS = emailrelay.sh
|
||||
pkgdata_DATA = emailrelay-notify.sh emailrelay-deliver.sh
|
||||
EXTRA_DIST = emailrelay-filter.sh_ emailrelay-test.sh_ emailrelay.sh_ txt2html.sh_ emailrelay-notify.sh_ emailrelay-deliver.sh_
|
||||
pkgdata_DATA = emailrelay-notify.sh emailrelay-deliver.sh emailrelay-process.sh
|
||||
EXTRA_DIST = emailrelay-doxygen-filter.sh_ emailrelay-test.sh_ emailrelay.sh_ txt2html.sh_ emailrelay-notify.sh_ emailrelay-deliver.sh_ emailrelay-process.sh_
|
||||
CLEANFILES = $(noinst_SCRIPTS) $(libexec_SCRIPTS) $(pkgdata_DATA)
|
||||
TESTS = emailrelay-test.sh
|
||||
SUFFIXES = .sh_ .sh
|
||||
|
@ -69,10 +69,10 @@ PACKAGE = @PACKAGE@
|
||||
RANLIB = @RANLIB@
|
||||
VERSION = @VERSION@
|
||||
|
||||
noinst_SCRIPTS = emailrelay-filter.sh emailrelay-test.sh
|
||||
noinst_SCRIPTS = emailrelay-doxygen-filter.sh emailrelay-test.sh
|
||||
libexec_SCRIPTS = emailrelay.sh
|
||||
pkgdata_DATA = emailrelay-notify.sh emailrelay-deliver.sh
|
||||
EXTRA_DIST = emailrelay-filter.sh_ emailrelay-test.sh_ emailrelay.sh_ txt2html.sh_ emailrelay-notify.sh_ emailrelay-deliver.sh_
|
||||
pkgdata_DATA = emailrelay-notify.sh emailrelay-deliver.sh emailrelay-process.sh
|
||||
EXTRA_DIST = emailrelay-doxygen-filter.sh_ emailrelay-test.sh_ emailrelay.sh_ txt2html.sh_ emailrelay-notify.sh_ emailrelay-deliver.sh_ emailrelay-process.sh_
|
||||
CLEANFILES = $(noinst_SCRIPTS) $(libexec_SCRIPTS) $(pkgdata_DATA)
|
||||
TESTS = emailrelay-test.sh
|
||||
SUFFIXES = .sh_ .sh
|
||||
|
@ -21,12 +21,22 @@
|
||||
#
|
||||
# emailrelay-deliver.sh
|
||||
#
|
||||
# An example script which looks for local mail in the MailRelay
|
||||
# An example script which looks for local mail in the E-MailRelay
|
||||
# spool directory, and delivers it using 'procmail'.
|
||||
#
|
||||
# usage: emailrelay-deliver.sh [<spool-dir>]
|
||||
#
|
||||
# Note that (1) E-MailRelay will only accept mail to a local recipient
|
||||
# if that recipient is "postmaster", and the (2) by default E-MailRelay
|
||||
# will only accept connections from local e-mail clients. This means that
|
||||
# this script is only relevant if you are in the habit of sending
|
||||
# yourself mail as "postmaster". It is provided for completeness,
|
||||
# in response to certain requirements in the SMTP specification.
|
||||
#
|
||||
|
||||
store="/var/spool/mailrelay"
|
||||
store="/usr/local/var/spool/emailrelay"
|
||||
postmaster="root"
|
||||
procmail="procmail"
|
||||
|
||||
# parse the command line
|
||||
#
|
||||
@ -45,7 +55,7 @@ fi
|
||||
|
||||
# for each e-mail to a local recipient...
|
||||
#
|
||||
for file in ${store}/mail-relay.*.envelope.local ""
|
||||
for file in ${store}/emailrelay.*.envelope.local ""
|
||||
do
|
||||
if test -f "${file}"
|
||||
then
|
||||
@ -62,11 +72,11 @@ do
|
||||
if test -f "${content}"
|
||||
then
|
||||
echo `basename $0`: delivering `basename ${content}` to ${deliver_to}
|
||||
procmail -d ${deliver_to} < ${content}
|
||||
${procmail} -d ${deliver_to} < ${content}
|
||||
rc=$?
|
||||
if test "${rc}" -eq 0
|
||||
then
|
||||
echo '' # rm -f "${file}" 2>/dev/null
|
||||
rm -f "${file}" 2>/dev/null
|
||||
fi
|
||||
fi
|
||||
fi
|
||||
|
@ -21,16 +21,30 @@
|
||||
#
|
||||
# emailrelay-notify.sh
|
||||
#
|
||||
# An example script which looks for failed mail in the MailRelay spool
|
||||
# An example script which looks for failed mail in the E-MailRelay spool
|
||||
# directory, and sends failure notification messages using 'procmail'.
|
||||
#
|
||||
# usage: emailrelay-notify.sh [<spool-dir>]
|
||||
#
|
||||
# Notification of failed e-mail by means of e-mail messages is a
|
||||
# requirement imposed by the SMTP specification. However, a simpler
|
||||
# approach might be more appropriate -- for example, a line like this in
|
||||
# a ".profile" script:
|
||||
#
|
||||
# if test -f /usr/local/var/spool/emailrelay/*.envelope.bad ; then echo Failed mail >&2 ; fi
|
||||
#
|
||||
# or perhaps a cron entry like this (since output from a cron job gets sent as mail):
|
||||
#
|
||||
# 0 0 * * * /bin/cat /usr/local/var/spool/emailrelay/*.envelope.bad 2>/dev/null
|
||||
#
|
||||
|
||||
tmp="/tmp/`basename $0`.$$.tmp"
|
||||
trap "rm -f ${tmp} 2>/dev/null ; exit 0" 0 1 2 3 13 15
|
||||
procmail="procmail"
|
||||
|
||||
# parse the command line
|
||||
#
|
||||
store="/var/spool/mailrelay"
|
||||
store="/usr/local/var/spool/emailrelay"
|
||||
if test $# -ge 1
|
||||
then
|
||||
store="${1}"
|
||||
@ -46,10 +60,12 @@ fi
|
||||
|
||||
# for each failed e-mail...
|
||||
#
|
||||
for file in ${store}/mail-relay.*.envelope.bad ""
|
||||
for file in ${store}/emailrelay.*.envelope.bad ""
|
||||
do
|
||||
if test -f "${file}"
|
||||
then
|
||||
# parse out the failure reason and the original sender
|
||||
#
|
||||
content="`echo ${file} | sed 's/envelope/content/' | sed 's/.bad//'`"
|
||||
reason="`fgrep MailRelay-Reason ${file} | sed 's/X-[^ ]*Reason: //' | tr -d '\015'`"
|
||||
from="`fgrep MailRelay-From ${file} | sed 's/X-MailRelay-From: //' | tr -d '\015'`"
|
||||
@ -61,33 +77,53 @@ do
|
||||
|
||||
# create a notification message header
|
||||
#
|
||||
boundary="--------------`basename ${file}`.$$"
|
||||
echo "To: ${deliver_to}" > ${tmp}
|
||||
echo "From: postmaster" >> ${tmp}
|
||||
echo "Subject: Your e-mail could not be delivered" >> ${tmp}
|
||||
echo " " >> ${tmp}
|
||||
echo "MIME-Version: 0.1" >> ${tmp}
|
||||
echo "Content-Type: multipart/mixed; boundary=\"${boundary}\"" >> ${tmp}
|
||||
echo "" >> ${tmp}
|
||||
|
||||
# add the message content
|
||||
# add the message text
|
||||
#
|
||||
echo "" >> ${tmp}
|
||||
echo "--${boundary}" >> ${tmp}
|
||||
echo "Content-Type: text/plain; charset=us-ascii" >> ${tmp}
|
||||
echo "" >> ${tmp}
|
||||
if test -f "${content}"
|
||||
then
|
||||
egrep -i '^To:|^Subject:' ${content} >> ${tmp}
|
||||
fi
|
||||
if test "${reason}" != ""
|
||||
then
|
||||
echo "Reason: ${reason}" >> ${tmp}
|
||||
fi
|
||||
|
||||
# add the message attachment
|
||||
#
|
||||
if test -f "${content}"
|
||||
then
|
||||
egrep -i '^To:|^Subject:' ${content} >> ${tmp}
|
||||
echo " " >> ${tmp}
|
||||
echo "The original mail is saved as \"${content}\"." >> ${tmp}
|
||||
echo "You should make a copy of this file if necessary and then delete it." >> ${tmp}
|
||||
echo "--${boundary}" >> ${tmp}
|
||||
echo "Content-Type: message/rfc822" >> ${tmp}
|
||||
echo "Content-Transfer-Encoding: 8bit" >> ${tmp}
|
||||
echo "Content-Description: `basename ${content}`" >> ${tmp}
|
||||
echo "" >> ${tmp}
|
||||
cat ${content} >> ${tmp}
|
||||
fi
|
||||
echo "--${boundary}--" >> ${tmp}
|
||||
|
||||
# deliver the notification using procmail
|
||||
#
|
||||
echo `basename $0`: delivering `basename ${content}` to ${deliver_to}
|
||||
procmail -d ${deliver_to} < ${tmp}
|
||||
${procmail} -d ${deliver_to} < ${tmp}
|
||||
rc=$?
|
||||
|
||||
# clean up
|
||||
#
|
||||
if test "${rc}" -eq 0
|
||||
then
|
||||
rm -f "${file}" 2>/dev/null
|
||||
rm -f "${file}" "${content}" 2>/dev/null
|
||||
fi
|
||||
fi
|
||||
done
|
||||
|
146
bin/emailrelay-process.sh_
Normal file
146
bin/emailrelay-process.sh_
Normal file
@ -0,0 +1,146 @@
|
||||
#!/bin/sh
|
||||
#
|
||||
# Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
#
|
||||
# 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, write to the Free Software
|
||||
# Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
#
|
||||
# ===
|
||||
#
|
||||
# emailrelay-process.sh
|
||||
#
|
||||
# An example pre-processing script for the E-MailRelay
|
||||
# system which does rot-13 masking.
|
||||
#
|
||||
|
||||
awk="awk"
|
||||
tmp="/tmp/`basename $0`.$$.tmp"
|
||||
log="/tmp/`basename $0`.$$.out"
|
||||
trap "rm -f ${tmp} >/dev/null 2>&1 ; exit" 0 1 2 3 13 15
|
||||
|
||||
###
|
||||
# ProcessContent()
|
||||
# Processes the content part of an RFC822 message. This
|
||||
# implementation does rot13 masking.
|
||||
#
|
||||
ProcessContent()
|
||||
{
|
||||
${awk} '
|
||||
BEGIN {
|
||||
map_upper = "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
|
||||
map_lower = tolower(map_upper)
|
||||
in_header = 1
|
||||
}
|
||||
|
||||
function rot( n , c , map )
|
||||
{
|
||||
return index(map,c) ? substr(map,((index(map,c)+n-1)%length(map))+1,1) : c
|
||||
}
|
||||
|
||||
function rot_c( n , c )
|
||||
{
|
||||
return rot(n,rot(n,c,map_upper),map_lower)
|
||||
}
|
||||
|
||||
function rot_s( n , string )
|
||||
{
|
||||
result = ""
|
||||
for( i = 1 ; i <= length(string) ; i++ )
|
||||
result = result rot_c(n,substr(string,i,1))
|
||||
return result
|
||||
}
|
||||
|
||||
{
|
||||
is_blank = match($0,"^[[:space:]]*$")
|
||||
if( in_header && is_blank )
|
||||
in_header = 0
|
||||
|
||||
if( in_header )
|
||||
print
|
||||
else
|
||||
print rot_s(13,$0)
|
||||
}
|
||||
'
|
||||
}
|
||||
|
||||
###
|
||||
# Wrap()
|
||||
# Processes an RCF822 message so that the original content
|
||||
# appears as an attachment.
|
||||
#
|
||||
Wrap()
|
||||
{
|
||||
${awk} -v boundary="-----`basename $0`.$$" -v message="$@" '
|
||||
BEGIN {
|
||||
in_header = 1
|
||||
n = 1
|
||||
}
|
||||
{
|
||||
is_blank = match($0,"^[[:space:]]*$")
|
||||
if( in_header && is_blank )
|
||||
{
|
||||
printf( "Content-Type: multipart/mixed; boundary=\"%s\"\r\n" , boundary )
|
||||
printf( "\r\n" )
|
||||
printf( "\r\n" )
|
||||
printf( "--%s\r\n" , boundary )
|
||||
printf( "Content-Type: text/plain; charset=us-ascii\r\n" )
|
||||
printf( "\r\n" )
|
||||
printf( "%s\r\n" , message )
|
||||
printf( "\r\n" )
|
||||
printf( "--%s\r\n" , boundary )
|
||||
printf( "Content-Type: message/rfc822\r\n" )
|
||||
printf( "Content-Transfer-Encoding: 8bit\r\n" )
|
||||
printf( "Content-Description: encrypted message\r\n" )
|
||||
printf( "\r\n" )
|
||||
for( i = 1 ; i < n ; i++ )
|
||||
print header[i]
|
||||
}
|
||||
|
||||
if( in_header && is_blank )
|
||||
in_header = 0
|
||||
|
||||
if( in_header )
|
||||
{
|
||||
header[n++] = $0
|
||||
|
||||
is_mime_content = match($0,"^Content-")
|
||||
is_continuation = match($0,"^[[:space:]][[:space:]]*[^[:space:]]")
|
||||
block = is_mime_content || (was_mime_content && is_continuation)
|
||||
was_mime_content = block
|
||||
|
||||
if( ! block )
|
||||
print
|
||||
}
|
||||
else
|
||||
{
|
||||
print
|
||||
}
|
||||
}
|
||||
END {
|
||||
printf( "--%s--\r\n" , boundary )
|
||||
printf( "\r\n" )
|
||||
}
|
||||
'
|
||||
}
|
||||
|
||||
Main()
|
||||
{
|
||||
cat "${1}" | ProcessContent | Wrap "The original message has been encrypted..." > ${tmp}
|
||||
cp ${tmp} "${1}"
|
||||
}
|
||||
|
||||
# Main $@ > ${log} 2>&1
|
||||
Main $@
|
||||
|
@ -23,11 +23,14 @@
|
||||
#
|
||||
# Test the E-MailRelay system.
|
||||
#
|
||||
# Creates three temporary spool directories under /tmp and runs
|
||||
# three emailrelay servers to bucket-brigade a test message from one to
|
||||
# the next. The test succeeds if the message gets into the third
|
||||
# Creates four temporary spool directories under /tmp and runs
|
||||
# four emailrelay servers to bucket-brigade a test message from one to
|
||||
# the next. The test succeeds if the message gets into the final
|
||||
# spool directory.
|
||||
#
|
||||
# Once all the servers have been killed the separate log files are
|
||||
# concatenated into the summary log file "/tmp/emailrelay-test.out".
|
||||
#
|
||||
# If this test takes more than a second or two then it has failed.
|
||||
#
|
||||
|
||||
@ -42,7 +45,8 @@ Cleanup()
|
||||
kill `cat ${base_dir}/pid-* 2>/dev/null` 2>/dev/null
|
||||
if test -d ${base_dir}
|
||||
then
|
||||
grep "." ${base_dir}/log-? > /tmp/`basename $0`.out 2>/dev/null
|
||||
grep "MailRelay-Reason" ${base_dir}/*/*envelope*bad > /tmp/`basename $0`.out 2>/dev/null
|
||||
grep "." ${base_dir}/log-? >> /tmp/`basename $0`.out 2>/dev/null
|
||||
fi
|
||||
rm -rf ${base_dir} 2>/dev/null
|
||||
}
|
||||
@ -113,7 +117,7 @@ CrLf()
|
||||
|
||||
CheckResults()
|
||||
{
|
||||
if test -f ${base_dir}/store-3/*.envelope -a -f ${base_dir}/store-3/*.content
|
||||
if test -f ${base_dir}/store-4/*.envelope -a -f ${base_dir}/store-4/*.content
|
||||
then
|
||||
exit_code="0"
|
||||
echo `basename $0`: succeeded
|
||||
@ -139,10 +143,11 @@ trap "Trap 0 ; exit" 0
|
||||
|
||||
StartTimer
|
||||
RunServer ${pp}1 store-2 log-1 pid-1
|
||||
RunServer ${pp}2 store-2 log-2 pid-2 "--admin ${pp}4 --forward-to localhost:${pp}3"
|
||||
RunServer ${pp}3 store-3 log-3 pid-3
|
||||
RunServer ${pp}2 store-2 log-2 pid-2 "--admin ${pp}9 --forward-to localhost:${pp}3"
|
||||
RunServer ${pp}3 store-3 log-3 pid-3 "--immediate --forward-to localhost:${pp}4 --filter /bin/touch"
|
||||
RunServer ${pp}4 store-4 log-4 pid-4
|
||||
CreateMessages
|
||||
RunClient localhost:${pp}1 store-1 log-c pid-4
|
||||
RunPoke ${pp}4 log-p
|
||||
RunClient localhost:${pp}1 store-1 log-c pid-5
|
||||
RunPoke ${pp}9 log-p
|
||||
CheckResults
|
||||
|
||||
|
@ -24,14 +24,19 @@
|
||||
# A shell-script wrapper for E-MailRelay designed for
|
||||
# use in the SysV-init system (/etc/init.d).
|
||||
#
|
||||
# usage: emailrelay.sh { start | stop }
|
||||
# usage: emailrelay.sh { start [<emailrelay-switches>] | stop }
|
||||
#
|
||||
|
||||
# configuration
|
||||
#
|
||||
var_run="/var/run"
|
||||
emailrelay="/usr/local/sbin/emailrelay"
|
||||
switches=""
|
||||
|
||||
# configuration fallback
|
||||
#
|
||||
if test \! -d "${var_run}" ; then var_run="/tmp" ; fi
|
||||
params=""
|
||||
if test \! -x "${emailrelay}" ; then emailrelay="emailrelay" ; fi
|
||||
|
||||
# initialisation
|
||||
#
|
||||
@ -52,8 +57,9 @@ if test "${1}" = "start"
|
||||
then
|
||||
# "start"
|
||||
#
|
||||
shift
|
||||
rm -f "${pid_file}" 2>/dev/null
|
||||
emailrelay --as-server --pid-file "${pid_file}" ${params} $@
|
||||
${emailrelay} --as-server --pid-file "${pid_file}" ${switches} $@
|
||||
|
||||
elif test "${1}" = "stop"
|
||||
then
|
||||
|
259
bin/txt2html.sh_
259
bin/txt2html.sh_
@ -21,10 +21,20 @@
|
||||
#
|
||||
# txt2html.sh
|
||||
#
|
||||
# usage: txt2html.sh [-a <awk-binary>] <input-file>
|
||||
# Converts plain-text to html. The plain-text has to use special
|
||||
# formating conventions (see "function process()", Anchorise_1() etc).
|
||||
#
|
||||
# Embeds comments in the html output which can be used by "index.sh"
|
||||
# to create a document index.
|
||||
#
|
||||
# Definition lists require a bullet graphic, "graphics/bullet.gif".
|
||||
#
|
||||
# usage: txt2html.sh [-a <awk-binary>] [-x] <input-file> [<title>]
|
||||
#
|
||||
# The -x switch excludes header and footer stuff.
|
||||
#
|
||||
|
||||
awk="awk"
|
||||
awk="gawk"
|
||||
if test "${1}" = "-a"
|
||||
then
|
||||
shift
|
||||
@ -35,6 +45,13 @@ then
|
||||
shift
|
||||
fi
|
||||
|
||||
full="1"
|
||||
if test "${1}" = "-x"
|
||||
then
|
||||
shift
|
||||
full="0"
|
||||
fi
|
||||
|
||||
file="${1}"
|
||||
if test "${file}" = ""
|
||||
then
|
||||
@ -48,23 +65,58 @@ then
|
||||
exit 1
|
||||
fi
|
||||
|
||||
title="`basename ${file}`"
|
||||
title="`grep -v '^[[:space:]]*$' ${file} | head -1`"
|
||||
if test "${2}" != ""
|
||||
then
|
||||
title="${2}"
|
||||
fi
|
||||
|
||||
# ===
|
||||
# Include()
|
||||
#
|
||||
# Expands #include# directives. Included text is then processed
|
||||
# as plain text, just like the top-level file. An include directive
|
||||
# within a line (ie. not on the lhs) is treated as an inline
|
||||
# sustitution, like shell backticks.
|
||||
#
|
||||
Include()
|
||||
{
|
||||
${awk} -v cat="${awk} '{print}'" '
|
||||
{
|
||||
line = $0
|
||||
if( match(line,"^#include#[^#]*#") )
|
||||
{
|
||||
path = substr(line,10,RLENGTH-10)
|
||||
system( cat " " path )
|
||||
}
|
||||
else
|
||||
{
|
||||
print line
|
||||
}
|
||||
}
|
||||
'
|
||||
}
|
||||
|
||||
# ===
|
||||
# Main()
|
||||
#
|
||||
# Does the bulk of the conversion.
|
||||
#
|
||||
Main()
|
||||
{
|
||||
${awk} -v title="${1}" '
|
||||
${awk} -v title="${1}" -v full="${2}" -v colour="${3}" '
|
||||
BEGIN {
|
||||
if( full )
|
||||
{
|
||||
dtd = "-//W3C//DTD HTML 4.01 Transitional//EN"
|
||||
printf( "<!DOCTYPE HTML PUBLIC \"%s\">\n" , dtd )
|
||||
printf( "<html>\n" )
|
||||
printf( "<head>\n" )
|
||||
printf( "<title>%s</title>\n" , title )
|
||||
printf( "</head>\n" )
|
||||
printf( "<body>\n" )
|
||||
printf( "<body bgcolor=\"%s\">\n" , colour )
|
||||
printf( "<!-- index:0::::%s -->\n" , title )
|
||||
}
|
||||
}
|
||||
|
||||
function escape( line )
|
||||
@ -103,33 +155,53 @@ function tagOutput( line , tag )
|
||||
|
||||
function process( line , next_ )
|
||||
{
|
||||
is_blank = match( line , "^[[:space:]]*$" )
|
||||
is_sub_para = match( line , "^[[:space:]][[:space:]][^[:space:]]" )
|
||||
is_code = match( line , "^[[:space:]]" ) && !is_sub_para
|
||||
is_heading = match( next_ , "^==*[[:space:]]*$" )
|
||||
is_sub_heading = match( next_ , "^--*[[:space:]]*$" )
|
||||
tab = " "
|
||||
is_blank = match( line , "^ *$" )
|
||||
is_heading = match( next_ , "^==* *$" )
|
||||
is_sub_heading = match( next_ , "^--* *$" )
|
||||
is_list_item = match( line , "^\\* " )
|
||||
is_definition_term = match( line , "^\\# " )
|
||||
is_definition_text = match( line , "^ [^- ]" )
|
||||
is_outer_list_item = match( line , "^+ " )
|
||||
is_inner_list_item = match( line , "^ - " )
|
||||
is_sub_list_item = match( line , "^ + " )
|
||||
is_numbered_item = match( line , "^\\([[:digit:]][[:digit:]]*\\)" )
|
||||
is_heading_line = match( line , "^==*[[:space:]]*$" )
|
||||
is_sub_heading_line = match( line , "^--*[[:space:]]*$" )
|
||||
is_heading_line = match( line , "^==* *$" )
|
||||
is_sub_heading_line = match( line , "^--* *$" )
|
||||
is_code = match( line , "^" tab )
|
||||
|
||||
if( is_blank )
|
||||
{
|
||||
printf( "<p><br>\n" )
|
||||
printf( "<br><br>\n" )
|
||||
}
|
||||
else if( is_code )
|
||||
{
|
||||
tagOutput( line , "pre" )
|
||||
}
|
||||
else if( is_sub_para )
|
||||
else if( is_definition_term )
|
||||
{
|
||||
tagOutput( line , "sub" )
|
||||
gsub( "^# " , "" , line )
|
||||
tagOutput( line , "dt" )
|
||||
}
|
||||
else if( is_definition_text )
|
||||
{
|
||||
tagOutput( line , "dd" )
|
||||
}
|
||||
else if( is_list_item )
|
||||
{
|
||||
gsub( "^\\* " , "" , line )
|
||||
tagOutput( line , "li" )
|
||||
}
|
||||
else if( is_outer_list_item )
|
||||
{
|
||||
gsub( "^+ " , "" , line )
|
||||
tagOutput( line , "Li" )
|
||||
}
|
||||
else if( is_inner_list_item )
|
||||
{
|
||||
gsub( "^ - " , "" , line )
|
||||
tagOutput( line , "lI" )
|
||||
}
|
||||
else if( is_numbered_item )
|
||||
{
|
||||
gsub( "^\\([[:digit:]][[:digit:]]*\\) " , "" , line )
|
||||
@ -137,11 +209,16 @@ function process( line , next_ )
|
||||
}
|
||||
else if( is_heading )
|
||||
{
|
||||
printf( "<h1>%s</h1>\n" , line )
|
||||
major += 1
|
||||
minor = 0
|
||||
printf( "<h1><a name=\"H_%d\">%s</h1>" , major , line )
|
||||
printf( "<!-- index:1:H:%d::%s -->\n" , major , line )
|
||||
}
|
||||
else if( is_sub_heading )
|
||||
{
|
||||
printf( "<h2>%s</h2>\n" , line )
|
||||
minor += 1
|
||||
printf( "<h2><a name=\"SH_%d_%d\">%s</h2>" , major , minor , line )
|
||||
printf( "<!-- index:2:SH:%d:%d:%s -->\n" , major , minor , line )
|
||||
}
|
||||
else if( !is_heading_line && !is_sub_heading_line )
|
||||
{
|
||||
@ -157,18 +234,33 @@ function process( line , next_ )
|
||||
|
||||
END {
|
||||
process( previous , "" )
|
||||
if( full )
|
||||
{
|
||||
printf( "</body>\n" )
|
||||
printf( "</html>\n" )
|
||||
}
|
||||
} '
|
||||
}
|
||||
|
||||
# ==
|
||||
|
||||
# ===
|
||||
# AugmentLists()
|
||||
#
|
||||
# Adds list begin/end tags around a set of list items.
|
||||
#
|
||||
AugmentLists()
|
||||
{
|
||||
${awk} -v item_tag="${1}" -v list_tag="${2}" '
|
||||
${awk} -v item_tag="${1}" -v list_tag="${2}" -v ignore_1_re="${3}" -v ignore_2_re="${4}" '
|
||||
{
|
||||
line = $0
|
||||
ignore_1 = length(ignore_1_re) && match( line , ignore_1_re )
|
||||
ignore_2 = length(ignore_2_re) && match( line , ignore_2_re )
|
||||
ignore = ignore_1 || ignore_2
|
||||
if( ignore )
|
||||
{
|
||||
print
|
||||
}
|
||||
else
|
||||
{
|
||||
is_list_item = match( line , "^<" item_tag ">.*</" item_tag ">$" )
|
||||
|
||||
if( is_list_item && !in_list )
|
||||
@ -178,11 +270,21 @@ ${awk} -v item_tag="${1}" -v list_tag="${2}" '
|
||||
|
||||
print
|
||||
in_list = is_list_item
|
||||
}
|
||||
} '
|
||||
}
|
||||
|
||||
# ==
|
||||
|
||||
# ===
|
||||
# Elide()
|
||||
#
|
||||
# Converts repeated lines of <foo>lineN</foo> into
|
||||
# <foo>
|
||||
# line1
|
||||
# line2
|
||||
# </foo>
|
||||
#
|
||||
# Useful for <pre> and <sub>.
|
||||
#
|
||||
Elide()
|
||||
{
|
||||
${awk} -v tag="${1}" '
|
||||
@ -205,21 +307,47 @@ ${awk} -v tag="${1}" '
|
||||
} '
|
||||
}
|
||||
|
||||
# ==
|
||||
# ===
|
||||
# Decorate()
|
||||
#
|
||||
# Adds additional stuff after a given opening tag.
|
||||
# The tag is expected to be at the start of the line.
|
||||
#
|
||||
Decorate()
|
||||
{
|
||||
${awk} -v tag="${1}" -v decoration="${2}" '
|
||||
{
|
||||
line = $0
|
||||
sub( "^<" tag ">" , "<" tag ">" decoration , line )
|
||||
print line
|
||||
} '
|
||||
}
|
||||
|
||||
# ===
|
||||
# Compress()
|
||||
#
|
||||
# Removes blank lines near to headings (etc).
|
||||
#
|
||||
Compress()
|
||||
{
|
||||
${awk} '
|
||||
function process( previous , line , next_ )
|
||||
{
|
||||
re_blank = "^<p><br>$"
|
||||
re_blank = "^<br><br>$"
|
||||
re_heading = "^<[Hh][[:digit:]]>"
|
||||
re_dd = "^<dd>"
|
||||
re_pre_start = "^<pre>"
|
||||
re_pre_end = "</pre>$"
|
||||
if( match(line,re_blank) && ( match(next_,re_heading) || match(previous,re_heading) ) )
|
||||
|
||||
this_is_blank = match(line,re_blank)
|
||||
next_is_special = match(next_,re_heading) || match(next_,re_dd)
|
||||
previous_is_special = match(previous,re_heading) || match(previous,re_dd)
|
||||
next_is_pre_start = match(next_,re_pre_start)
|
||||
|
||||
if( this_is_blank && ( next_is_special || previous_is_special ) )
|
||||
{
|
||||
}
|
||||
else if( match(line,re_blank) && match(next_,re_pre_start) )
|
||||
else if( this_is_blank && next_is_pre_start )
|
||||
{
|
||||
}
|
||||
else
|
||||
@ -240,7 +368,84 @@ END {
|
||||
'
|
||||
}
|
||||
|
||||
# ===
|
||||
# Anchorise_1()
|
||||
#
|
||||
# Converts [[-foo-bar-]] to <a href="foo">bar</a>.
|
||||
#
|
||||
Anchorise_1()
|
||||
{
|
||||
sed 's/\[\[-\([^-]*\)-\([^-]*\)-\]\]/<a href="\1">\2<\/a>/g'
|
||||
}
|
||||
|
||||
# ===
|
||||
# Anchorise_2()
|
||||
#
|
||||
# Converts [[foo]] to <a href="../foo">foo</a>.
|
||||
#
|
||||
Anchorise_2()
|
||||
{
|
||||
sed 's/\[\[\([^\]*\)\]\]/<a href=..\/"\1">\1<\/a>/g'
|
||||
}
|
||||
|
||||
# ===
|
||||
# Cat()
|
||||
#
|
||||
# An awk version of "cat" (cygwin cat is
|
||||
# broken).
|
||||
#
|
||||
Cat()
|
||||
{
|
||||
${awk} '{print}' $@ | tr -d '\015'
|
||||
}
|
||||
|
||||
# ===
|
||||
# MoveIndex()
|
||||
#
|
||||
# Moves the index comments to a line before
|
||||
# the header, rather than at the end of the
|
||||
# header line.
|
||||
#
|
||||
MoveIndex()
|
||||
{
|
||||
${awk} '
|
||||
{
|
||||
line = $0
|
||||
re = "<!-- index:"
|
||||
pos = match(line,re)
|
||||
if( pos )
|
||||
{
|
||||
head = substr(line,1,pos-1)
|
||||
tail = substr(line,pos+length(re))
|
||||
print head
|
||||
print re tail
|
||||
}
|
||||
else
|
||||
{
|
||||
print line
|
||||
}
|
||||
}
|
||||
'
|
||||
}
|
||||
|
||||
# ==
|
||||
|
||||
cat "${file}" | Main "${title}" | Compress | AugmentLists li bl | AugmentLists LI ol | Elide "sub" | Elide "pre"
|
||||
colour="#FFFFFF"
|
||||
Cat "${file}" | \
|
||||
Include | \
|
||||
Main "${title}" "${full}" "${colour}" | \
|
||||
Compress | \
|
||||
AugmentLists li ul | \
|
||||
AugmentLists Li ul "^<lI>" "^<br><br>" | \
|
||||
AugmentLists lI ul | \
|
||||
AugmentLists LI ol | \
|
||||
AugmentLists dt dl "^<dd>" "^<br><br>" | \
|
||||
Elide "sub" | \
|
||||
Elide "dd" | \
|
||||
Elide "pre" | \
|
||||
Decorate dt "<img src=\"graphics/bullet.gif\">\\\ " | \
|
||||
Decorate dd "<p>" | \
|
||||
Anchorise_1 | \
|
||||
Anchorise_2 | \
|
||||
MoveIndex
|
||||
|
||||
|
@ -33,6 +33,9 @@
|
||||
/* Version number of package */
|
||||
#undef VERSION
|
||||
|
||||
/* auto_ptr assignment has non-const rhs */
|
||||
#undef HAVE_NONCONST_AUTOPTR
|
||||
|
||||
/* have reentrant localtime */
|
||||
#undef HAVE_LOCALTIME_R
|
||||
|
||||
|
77
configure
vendored
77
configure
vendored
@ -691,7 +691,7 @@ fi
|
||||
|
||||
PACKAGE=emailrelay
|
||||
|
||||
VERSION=0.9.2
|
||||
VERSION=0.9.3
|
||||
|
||||
if test "`cd $srcdir && pwd`" != "`pwd`" && test -f $srcdir/config.status; then
|
||||
{ echo "configure: error: source directory already configured; run "make distclean" there first" 1>&2; exit 1; }
|
||||
@ -2008,8 +2008,78 @@ else
|
||||
fi
|
||||
done
|
||||
|
||||
ac_ext=C
|
||||
# CXXFLAGS is not in ac_cpp because -g, -O, etc. are not valid cpp options.
|
||||
ac_cpp='$CXXCPP $CPPFLAGS'
|
||||
ac_compile='${CXX-g++} -c $CXXFLAGS $CPPFLAGS conftest.$ac_ext 1>&5'
|
||||
ac_link='${CXX-g++} -o conftest${ac_exeext} $CXXFLAGS $CPPFLAGS $LDFLAGS conftest.$ac_ext $LIBS 1>&5'
|
||||
cross_compiling=$ac_cv_prog_cxx_cross
|
||||
|
||||
cat > conftest.$ac_ext <<EOF
|
||||
#line 2013 "configure"
|
||||
#line 2020 "configure"
|
||||
#include "confdefs.h"
|
||||
#include <memory>
|
||||
int main() {
|
||||
std::auto_ptr<int> lhs ; const std::auto_ptr<int> & rhs = lhs ; lhs = rhs ;
|
||||
; return 0; }
|
||||
EOF
|
||||
if { (eval echo configure:2027: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then
|
||||
:
|
||||
else
|
||||
echo "configure: failed program was:" >&5
|
||||
cat conftest.$ac_ext >&5
|
||||
rm -rf conftest*
|
||||
cat >> confdefs.h <<\EOF
|
||||
#define HAVE_NONCONST_AUTOPTR 1
|
||||
EOF
|
||||
|
||||
fi
|
||||
rm -f conftest*
|
||||
echo $ac_n "checking how to run the C++ preprocessor""... $ac_c" 1>&6
|
||||
echo "configure:2040: checking how to run the C++ preprocessor" >&5
|
||||
if test -z "$CXXCPP"; then
|
||||
if eval "test \"`echo '$''{'ac_cv_prog_CXXCPP'+set}'`\" = set"; then
|
||||
echo $ac_n "(cached) $ac_c" 1>&6
|
||||
else
|
||||
ac_ext=C
|
||||
# CXXFLAGS is not in ac_cpp because -g, -O, etc. are not valid cpp options.
|
||||
ac_cpp='$CXXCPP $CPPFLAGS'
|
||||
ac_compile='${CXX-g++} -c $CXXFLAGS $CPPFLAGS conftest.$ac_ext 1>&5'
|
||||
ac_link='${CXX-g++} -o conftest${ac_exeext} $CXXFLAGS $CPPFLAGS $LDFLAGS conftest.$ac_ext $LIBS 1>&5'
|
||||
cross_compiling=$ac_cv_prog_cxx_cross
|
||||
CXXCPP="${CXX-g++} -E"
|
||||
cat > conftest.$ac_ext <<EOF
|
||||
#line 2053 "configure"
|
||||
#include "confdefs.h"
|
||||
#include <stdlib.h>
|
||||
EOF
|
||||
ac_try="$ac_cpp conftest.$ac_ext >/dev/null 2>conftest.out"
|
||||
{ (eval echo configure:2058: \"$ac_try\") 1>&5; (eval $ac_try) 2>&5; }
|
||||
ac_err=`grep -v '^ *+' conftest.out | grep -v "^conftest.${ac_ext}\$"`
|
||||
if test -z "$ac_err"; then
|
||||
:
|
||||
else
|
||||
echo "$ac_err" >&5
|
||||
echo "configure: failed program was:" >&5
|
||||
cat conftest.$ac_ext >&5
|
||||
rm -rf conftest*
|
||||
CXXCPP=/lib/cpp
|
||||
fi
|
||||
rm -f conftest*
|
||||
ac_cv_prog_CXXCPP="$CXXCPP"
|
||||
ac_ext=C
|
||||
# CXXFLAGS is not in ac_cpp because -g, -O, etc. are not valid cpp options.
|
||||
ac_cpp='$CXXCPP $CPPFLAGS'
|
||||
ac_compile='${CXX-g++} -c $CXXFLAGS $CPPFLAGS conftest.$ac_ext 1>&5'
|
||||
ac_link='${CXX-g++} -o conftest${ac_exeext} $CXXFLAGS $CPPFLAGS $LDFLAGS conftest.$ac_ext $LIBS 1>&5'
|
||||
cross_compiling=$ac_cv_prog_cxx_cross
|
||||
fi
|
||||
fi
|
||||
CXXCPP="$ac_cv_prog_CXXCPP"
|
||||
echo "$ac_t""$CXXCPP" 1>&6
|
||||
|
||||
cat > conftest.$ac_ext <<EOF
|
||||
#line 2083 "configure"
|
||||
#include "confdefs.h"
|
||||
#include <time.h>
|
||||
EOF
|
||||
@ -2024,7 +2094,7 @@ fi
|
||||
rm -f conftest*
|
||||
|
||||
cat > conftest.$ac_ext <<EOF
|
||||
#line 2028 "configure"
|
||||
#line 2098 "configure"
|
||||
#include "confdefs.h"
|
||||
#include <time.h>
|
||||
EOF
|
||||
@ -2193,6 +2263,7 @@ s%@AR@%$AR%g
|
||||
s%@HAVE_DOXYGEN@%$HAVE_DOXYGEN%g
|
||||
s%@HAVE_MAN2HTML@%$HAVE_MAN2HTML%g
|
||||
s%@CPP@%$CPP%g
|
||||
s%@CXXCPP@%$CXXCPP%g
|
||||
|
||||
CEOF
|
||||
EOF
|
||||
|
23
configure.in
23
configure.in
@ -17,10 +17,29 @@ dnl along with this program; if not, write to the Free Software
|
||||
dnl Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
dnl
|
||||
dnl ===
|
||||
dnl
|
||||
dnl Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
dnl
|
||||
dnl This program is free software; you can redistribute it and/or
|
||||
dnl modify it under the terms of the GNU General Public License
|
||||
dnl as published by the Free Software Foundation; either
|
||||
dnl version 2 of the License, or (at your option) any later
|
||||
dnl version.
|
||||
dnl
|
||||
dnl This program is distributed in the hope that it will be useful,
|
||||
dnl but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
dnl MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
dnl GNU General Public License for more details.
|
||||
dnl
|
||||
dnl You should have received a copy of the GNU General Public License
|
||||
dnl along with this program; if not, write to the Free Software
|
||||
dnl Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
dnl
|
||||
dnl ===
|
||||
dnl Process this file with autoconf to produce a configure script.
|
||||
|
||||
AC_INIT(src/main/gsmtp.h)
|
||||
AM_INIT_AUTOMAKE(emailrelay,0.9.2)
|
||||
AM_INIT_AUTOMAKE(emailrelay,0.9.3)
|
||||
AM_CONFIG_HEADER(config.h)
|
||||
|
||||
dnl ===
|
||||
@ -52,6 +71,8 @@ AC_HEADER_TIME
|
||||
AC_CHECK_HEADERS(unistd.h)
|
||||
AC_CHECK_HEADERS(sys/time.h)
|
||||
AC_CHECK_FUNCS(glob)
|
||||
AC_LANG_CPLUSPLUS
|
||||
AC_TRY_COMPILE([#include <memory>],std::auto_ptr<int> lhs ; const std::auto_ptr<int> & rhs = lhs ; lhs = rhs ;,,AC_DEFINE(HAVE_NONCONST_AUTOPTR,1,auto_ptr assignment has non-const rhs))
|
||||
dnl check for *_r() declarations -- using ac_check_funcs
|
||||
dnl is no good here since they may be in the library but
|
||||
dnl not the header (depending on preprocessor switches)
|
||||
|
32
doc/Makefile.am
Normal file → Executable file
32
doc/Makefile.am
Normal file → Executable file
@ -1,32 +1,36 @@
|
||||
EXTRA_DIST = developer.txt reference.txt userguide.txt index.html emailrelay.1 emailrelay-poke.1 index.html
|
||||
EXTRA_DIST = developer.txt reference.txt userguide.txt index.html emailrelay.1 emailrelay-poke.1 doxygen_header.html graphics/bullet.gif
|
||||
|
||||
noinst_SCRIPTS = .dox
|
||||
pkgdata_DATA = readme.html developer.html reference.html userguide.html man.html index.html
|
||||
CLEANFILES = $(pkgdata_DATA) $(noinst_SCRIPTS) html
|
||||
CLEANFILES = $(noinst_SCRIPTS) html *.ht readme.html developer.html reference.html userguide.html man.html
|
||||
|
||||
SUFFIXES = .txt .html
|
||||
SUFFIXES = .txt .html .ht
|
||||
|
||||
filter=$(top_builddir)/bin/emailrelay-filter.sh
|
||||
filter_src=$(top_srcdir)/bin/emailrelay-filter.sh_
|
||||
filter=$(top_builddir)/bin/emailrelay-doxygen-filter.sh
|
||||
filter_src=$(top_srcdir)/bin/emailrelay-doxygen-filter.sh_
|
||||
converter=$(top_builddir)/bin/txt2html.sh
|
||||
converter_src=$(top_srcdir)/bin/txt2html.sh_
|
||||
|
||||
.txt.html:
|
||||
$(converter) -a "$(AWK)" $< > $*.html
|
||||
|
||||
.txt.ht:
|
||||
$(converter) -a "$(AWK)" -x $< > $*.ht
|
||||
|
||||
$(filter): $(filter_src)
|
||||
cp $(filter_src) $(filter)
|
||||
chmod ugo+x $(filter)
|
||||
|
||||
$(converter): $(converter_src)
|
||||
cp $(converter_src) $(converter)
|
||||
chmod ugo+x $(converter)
|
||||
|
||||
.dox: $(filter)
|
||||
if test "$(HAVE_DOXYGEN)" = "yes" ; then cat $(top_srcdir)/src/main/doxygen.cfg | sed "s:__TOP_SRC__:$(top_srcdir):g" | sed "s:__TOP_BUILD__:$(top_builddir):g" | doxygen - && touch .dox ; else echo no doxygen ; fi
|
||||
|
||||
man.html: emailrelay.1
|
||||
if test "$(HAVE_MAN2HTML)" = "yes" ; then man2html emailrelay.1 > man.html ; else echo no man2html ; fi
|
||||
|
||||
$(filter): $(filter_src)
|
||||
cp $< $(filter)
|
||||
chmod ugo+x $(filter)
|
||||
|
||||
$(converter): $(converter_src)
|
||||
cp $< $(converter)
|
||||
chmod ugo+x $(converter)
|
||||
|
||||
developer.html reference.html userguide.html: $(converter)
|
||||
|
||||
readme.html: $(top_srcdir)/README $(converter)
|
||||
@ -36,6 +40,8 @@ install-data-local:
|
||||
$(mkinstalldirs) $(destdir)$(mandir)/man1
|
||||
$(INSTALL) $(top_srcdir)/doc/emailrelay.1 $(destdir)$(mandir)/man1/emailrelay.1
|
||||
$(INSTALL) $(top_srcdir)/doc/emailrelay-poke.1 $(destdir)$(mandir)/man1/emailrelay-poke.1
|
||||
$(mkinstalldirs) $(destdir)$(pkgdatadir)/graphics
|
||||
$(INSTALL) $(top_srcdir)/doc/graphics/bullet.gif $(destdir)$(pkgdatadir)/graphics/bullet.gif
|
||||
$(mkinstalldirs) $(destdir)$(pkgdatadir)/html
|
||||
if test "$(HAVE_DOXYGEN)" = "yes" ; then for file in html/* ; do $(INSTALL) $$file $(destdir)$(pkgdatadir)/$$file ; done ; fi
|
||||
|
||||
|
@ -69,15 +69,16 @@ PACKAGE = @PACKAGE@
|
||||
RANLIB = @RANLIB@
|
||||
VERSION = @VERSION@
|
||||
|
||||
EXTRA_DIST = developer.txt reference.txt userguide.txt index.html emailrelay.1 emailrelay-poke.1 index.html
|
||||
EXTRA_DIST = developer.txt reference.txt userguide.txt index.html emailrelay.1 emailrelay-poke.1 doxygen_header.html graphics/bullet.gif
|
||||
|
||||
noinst_SCRIPTS = .dox
|
||||
pkgdata_DATA = readme.html developer.html reference.html userguide.html man.html index.html
|
||||
CLEANFILES = $(pkgdata_DATA) $(noinst_SCRIPTS) html
|
||||
CLEANFILES = $(noinst_SCRIPTS) html *.ht readme.html developer.html reference.html userguide.html man.html
|
||||
|
||||
SUFFIXES = .txt .html
|
||||
SUFFIXES = .txt .html .ht
|
||||
|
||||
filter = $(top_builddir)/bin/emailrelay-filter.sh
|
||||
filter_src = $(top_srcdir)/bin/emailrelay-filter.sh_
|
||||
filter = $(top_builddir)/bin/emailrelay-doxygen-filter.sh
|
||||
filter_src = $(top_srcdir)/bin/emailrelay-doxygen-filter.sh_
|
||||
converter = $(top_builddir)/bin/txt2html.sh
|
||||
converter_src = $(top_srcdir)/bin/txt2html.sh_
|
||||
mkinstalldirs = $(SHELL) $(top_srcdir)/mkinstalldirs
|
||||
@ -100,7 +101,7 @@ TAR = tar
|
||||
GZIP_ENV = --best
|
||||
all: all-redirect
|
||||
.SUFFIXES:
|
||||
.SUFFIXES: .html .txt
|
||||
.SUFFIXES: .ht .html .txt
|
||||
$(srcdir)/Makefile.in: Makefile.am $(top_srcdir)/configure.in $(ACLOCAL_M4)
|
||||
cd $(top_srcdir) && $(AUTOMAKE) --gnu --include-deps doc/Makefile
|
||||
|
||||
@ -136,6 +137,7 @@ distdir = $(top_builddir)/$(PACKAGE)-$(VERSION)/$(subdir)
|
||||
subdir = doc
|
||||
|
||||
distdir: $(DISTFILES)
|
||||
$(mkinstalldirs) $(distdir)/graphics
|
||||
@for file in $(DISTFILES); do \
|
||||
d=$(srcdir); \
|
||||
if test -d $$d/$$file; then \
|
||||
@ -213,20 +215,23 @@ maintainer-clean
|
||||
.txt.html:
|
||||
$(converter) -a "$(AWK)" $< > $*.html
|
||||
|
||||
.txt.ht:
|
||||
$(converter) -a "$(AWK)" -x $< > $*.ht
|
||||
|
||||
$(filter): $(filter_src)
|
||||
cp $(filter_src) $(filter)
|
||||
chmod ugo+x $(filter)
|
||||
|
||||
$(converter): $(converter_src)
|
||||
cp $(converter_src) $(converter)
|
||||
chmod ugo+x $(converter)
|
||||
|
||||
.dox: $(filter)
|
||||
if test "$(HAVE_DOXYGEN)" = "yes" ; then cat $(top_srcdir)/src/main/doxygen.cfg | sed "s:__TOP_SRC__:$(top_srcdir):g" | sed "s:__TOP_BUILD__:$(top_builddir):g" | doxygen - && touch .dox ; else echo no doxygen ; fi
|
||||
|
||||
man.html: emailrelay.1
|
||||
if test "$(HAVE_MAN2HTML)" = "yes" ; then man2html emailrelay.1 > man.html ; else echo no man2html ; fi
|
||||
|
||||
$(filter): $(filter_src)
|
||||
cp $< $(filter)
|
||||
chmod ugo+x $(filter)
|
||||
|
||||
$(converter): $(converter_src)
|
||||
cp $< $(converter)
|
||||
chmod ugo+x $(converter)
|
||||
|
||||
developer.html reference.html userguide.html: $(converter)
|
||||
|
||||
readme.html: $(top_srcdir)/README $(converter)
|
||||
@ -236,6 +241,8 @@ install-data-local:
|
||||
$(mkinstalldirs) $(destdir)$(mandir)/man1
|
||||
$(INSTALL) $(top_srcdir)/doc/emailrelay.1 $(destdir)$(mandir)/man1/emailrelay.1
|
||||
$(INSTALL) $(top_srcdir)/doc/emailrelay-poke.1 $(destdir)$(mandir)/man1/emailrelay-poke.1
|
||||
$(mkinstalldirs) $(destdir)$(pkgdatadir)/graphics
|
||||
$(INSTALL) $(top_srcdir)/doc/graphics/bullet.gif $(destdir)$(pkgdatadir)/graphics/bullet.gif
|
||||
$(mkinstalldirs) $(destdir)$(pkgdatadir)/html
|
||||
if test "$(HAVE_DOXYGEN)" = "yes" ; then for file in html/* ; do $(INSTALL) $$file $(destdir)$(pkgdatadir)/$$file ; done ; fi
|
||||
|
||||
|
@ -1,5 +1,5 @@
|
||||
E-MailRelay Developer reference
|
||||
===============================
|
||||
E-MailRelay Developer guide
|
||||
===========================
|
||||
|
||||
Module structure
|
||||
----------------
|
||||
@ -10,18 +10,73 @@ network classes using the Berkley socket and Winsock APIs. Both libraries
|
||||
are portable between POSIX-like systems (eg. Linux) and Windows.
|
||||
|
||||
The application-level classes are implemented within the "GSmtp" namespace.
|
||||
The key classes in this namespace are "ClientProtocol", "ServerProtocol" and
|
||||
"MessageStore". The protocol and message-store functionality is brought
|
||||
together by the high-level "GSmtp::Server" and "GSmtp::Client" classes.
|
||||
The key interfaces in this namespace are "ClientProtocol", "ServerProtocol" and
|
||||
"MessageStore".
|
||||
|
||||
Under Windows the "gnet" library needs to create hidden GUI windows in
|
||||
order to receive network events. (Windows has historically built network
|
||||
event processing on top of the GUI event system, rather then implementing
|
||||
both GUI and network event handling on top of a generic event notification
|
||||
system as POSIX systems do.) The Windows GUI and event classes are put into
|
||||
a separate "src/win32" directory.
|
||||
Under Windows there is an additional library for event handling. Windows has
|
||||
historically built network event processing on top of the GUI event system,
|
||||
rather then implementing network events using a generic event notification
|
||||
system as POSIX systems do. This means that the "gnet" library has to be able
|
||||
to create GUI windows in order to process network events. The extra GUI and
|
||||
event classes are put into a separate library in the "src/win32" directory,
|
||||
using the namespace "GGui".
|
||||
|
||||
For a quick tour of the code look at the following headers:
|
||||
Class structure overview
|
||||
------------------------
|
||||
The message-store functionality uses three abstract interfaces: "MessageStore",
|
||||
"NewMessage" and "StoredMessage". The "NewMessage" interface is used to create
|
||||
messages within the store, and the "StoredMessage" interface is used for
|
||||
reading and extracting messages from the store. The concrete implementation
|
||||
classes based on these interfaces are respectively "FileStore", "NewFile" and
|
||||
"StoredFile".
|
||||
|
||||
The interaction betweeen the server protocol class and the message store is
|
||||
mediated by the "ProtocolMessage" interface. Two implementations of this
|
||||
interface are supplied: one for normal spooling ("ProtocolMessageStore"), and
|
||||
another for immediate forwarding ("ProtocolMessageForward").
|
||||
|
||||
The protocol and message-store functionality are brought together by the
|
||||
high-level "GSmtp::Server" and "GSmtp::Client" classes.
|
||||
|
||||
Directory structure
|
||||
-------------------
|
||||
|
||||
# src
|
||||
|
||||
Parent directory for source code.
|
||||
|
||||
# src/glib
|
||||
|
||||
A low-level class library, including classes for file-system abstraction,
|
||||
date and time, string utility functions, logging, command line parsing etc.
|
||||
|
||||
# src/gnet
|
||||
|
||||
A network library using Berkley sockets or Winsock.
|
||||
|
||||
# src/win32
|
||||
|
||||
Additional classes for windows event processing.
|
||||
|
||||
# src/main
|
||||
|
||||
Application-level classes for E-MailRelay.
|
||||
|
||||
# lib
|
||||
|
||||
Parent directory for ANSI C++ fixes
|
||||
|
||||
# lib/gcc2.95
|
||||
|
||||
Standard headers which are missing in gcc2.95
|
||||
|
||||
# lib/msvc6.0
|
||||
|
||||
Standard headers which are missing (or broken) in msvc6.0
|
||||
|
||||
Tour
|
||||
----
|
||||
For a quick bottom-up tour of the code take a look at the following headers:
|
||||
* src/glib/gpath.h
|
||||
* src/glib/gstr.h
|
||||
* lib/gcc2.95/sstream
|
||||
@ -33,43 +88,6 @@ For a quick tour of the code look at the following headers:
|
||||
* src/main/gserverprotocol.h
|
||||
* src/main/gsmtpserver.h
|
||||
|
||||
|
||||
Directory structure
|
||||
-------------------
|
||||
|
||||
* src
|
||||
|
||||
Parent directory for source code.
|
||||
|
||||
* src/glib
|
||||
|
||||
A low-level class library, including classes for file-system abstraction,
|
||||
date and time, string utility functions, logging, command line parsing etc.
|
||||
|
||||
* src/gnet
|
||||
|
||||
A network library using Berkley sockets or Winsock.
|
||||
|
||||
* src/win32
|
||||
|
||||
Additional classes for windows event processing.
|
||||
|
||||
* src/main
|
||||
|
||||
Application-level classes for E-MailRelay.
|
||||
|
||||
* lib
|
||||
|
||||
Parent directory for ANSI C++ fixes
|
||||
|
||||
* lib/gcc2.95
|
||||
|
||||
Standard headers which are missing in gcc2.95
|
||||
|
||||
* lib/msvc6.0
|
||||
|
||||
Standard headers which are missing (or broken) in msvc6.0
|
||||
|
||||
Portability
|
||||
-----------
|
||||
The E-MailRelay code is written in ANSI C++, using the following
|
||||
@ -89,39 +107,39 @@ but not:
|
||||
* covariant return
|
||||
* "mutable"
|
||||
|
||||
The header files "gdef.h" in "src/glib", and "gnet.h" in "src/gnet" are
|
||||
intended to be used to fix up portability issues such as missing standard
|
||||
types, non-standard system headers etc. Conditional compilation is not
|
||||
used outside of these headers (with the odd exception).
|
||||
The header files "gdef.h" in "src/glib", and "gnet.h" in "src/gnet" are intended
|
||||
to be used to fix up portability issues such as missing standard types,
|
||||
non-standard system headers etc. Conditional compilation is not used outside
|
||||
of these headers (with the odd exception).
|
||||
|
||||
Deficiencies in the ANSI C++ headers files provided by the compiler
|
||||
are fixed up in the "lib" directory tree. For example, the msvc6.0
|
||||
compiler sometimes does not put its names into the "std" namespace,
|
||||
even though the std-namespace headers are used. This can be worked round
|
||||
by additional "using" declarations in the "lib/msvc6.0" headers.
|
||||
These work-rounds are kept out of the "src" tree because they are likely
|
||||
to be fixed in later compiler releases. Standards-compliant compilers
|
||||
should not need to include any headers from the "lib" directory tree.
|
||||
Deficiencies in the ANSI C++ headers files provided by the compiler are fixed up
|
||||
in the "lib" directory tree. For example, the msvc6.0 compiler sometimes does
|
||||
not put its names into the "std" namespace, even though the std-namespace
|
||||
headers are used. This can be worked round by additional "using" declarations
|
||||
in the "lib/msvc6.0" headers. These work-rounds are kept out of the "src" tree
|
||||
because they are likely to be fixed in later compiler releases.
|
||||
Standards-compliant compilers should not need to include any headers from the
|
||||
"lib" directory tree.
|
||||
|
||||
Windows/unix portability is generally addressed by providing a common
|
||||
class declaration with two implementations. Where necessary a pimple
|
||||
pattern is used to hide the system-specific parts of the declaration.
|
||||
Windows/unix portability is generally addressed by providing a common class
|
||||
declaration with two implementations. Where necessary a "pimple" pattern is used
|
||||
to hide the system-specific parts of the declaration.
|
||||
|
||||
A good example is the "GDirectory" class used for iterating through files
|
||||
in a directory. The header file "src/glib/gdirectory.h" is common to both
|
||||
systems, but two implementations are provided in "gdirectory_unix.cpp"
|
||||
and "gdirectory_win32.cpp". The unix implementation uses opendir() and
|
||||
glob(), while the windows implementation uses FindFirstFile().
|
||||
A good example is the "G::Directory" class used for iterating through files in
|
||||
a directory. The header file "src/glib/gdirectory.h" is common to both systems,
|
||||
but two implementations are provided in "gdirectory_unix.cpp" and
|
||||
"gdirectory_win32.cpp". The unix implementation uses opendir() and glob(),
|
||||
while the windows implementation uses FindFirstFile().
|
||||
|
||||
Sometimes only parts of the implementation are system-specific. In these
|
||||
cases there are three source files per header. For example, "gsocket.cpp",
|
||||
Sometimes only small parts of the implementation are system-specific. In
|
||||
these cases there are three source files per header. For example, "gsocket.cpp",
|
||||
"gsocket_win32.cpp" and "gsocket_unix.cpp" in the "src/gnet" directory.
|
||||
|
||||
Porting
|
||||
-------
|
||||
If trying a port using a good ANSI C++ compiler then start by removing
|
||||
files from the "lib/gcc2.95" directory (or edit makefiles to remove the
|
||||
include path), and then review the following header files: "src/glib/gdef.h",
|
||||
Porting to other compilers
|
||||
--------------------------
|
||||
If trying a port using a good ANSI C++ compiler then start by removing files
|
||||
from the "lib/gcc2.95" directory (or edit makefiles to remove the include path),
|
||||
and then review the following header files: "src/glib/gdef.h",
|
||||
"src/gnet/gnet.h", "src/glib/gmemory.h".
|
||||
|
||||
The unix (ie. POSIX-like) implementation of the directory iteration class
|
||||
@ -132,29 +150,89 @@ moved into the message store class.
|
||||
|
||||
IPv6
|
||||
----
|
||||
IPv6 is supported at compile-time by selecting source files in the
|
||||
"src/gnet" directory ending "_ipv6.cpp" rather than "_ipv4.cpp".
|
||||
The code should be regarded as experimental.
|
||||
|
||||
Doxygen
|
||||
-------
|
||||
The commenting style used in header files is compatible with Doxygen
|
||||
if passed through the simple awk-based preprocessor "emailrelay-filter.sh".
|
||||
A "make" in the "doc" directory will run Doxygen if it is found on
|
||||
your path.
|
||||
IPv6 is supported at compile-time by selecting source files in the "src/gnet"
|
||||
directory ending "_ipv6.cpp" rather than "_ipv4.cpp". The code has been tested
|
||||
to a limited extent on Linux.
|
||||
|
||||
Windows build
|
||||
-------------
|
||||
A simple project file "emailrelay.dsp" for msvc6.0 is provided in the
|
||||
"src/main" directory.
|
||||
A simple project file "emailrelay.dsp" for msvc6.0 is provided in the "src/main"
|
||||
directory.
|
||||
|
||||
Style
|
||||
-----
|
||||
Tabs are used for indenting, not multiple spaces, but only at the left
|
||||
hand edge, and never within a line. This allows the reader to choose
|
||||
how deep the indenting should be (appropriate to their window size) by
|
||||
setting the editor's tabstop. Using spaces does not allow the reader this
|
||||
freedom.
|
||||
The commenting style used in header files is compatible with doxygen if passed
|
||||
through the simple awk-based preprocessor "emailrelay-doxygen-filter.sh". A
|
||||
"make" in the "doc" directory will run doxygen if it is found on your path.
|
||||
|
||||
Patterns
|
||||
--------
|
||||
Gang-of-four Design Patterns (ISBN 0-201-63361-2):
|
||||
|
||||
+ Factory method
|
||||
|
||||
- GNet::EventSources::create()
|
||||
- GNet::Server::newPeer()
|
||||
- GSmtp::MessageStore::newMessage()
|
||||
|
||||
+ Iterator
|
||||
|
||||
- G::DirectoryIterator
|
||||
- GNet::EventHandlerList::begin()/end()
|
||||
- GSmtp::MessageStore::iterator()
|
||||
|
||||
+ Singleton
|
||||
|
||||
- G::LogOutput
|
||||
- GGui::ApplicationInstance
|
||||
- GNet::EventSources
|
||||
- GSmtp::MessageStore
|
||||
|
||||
+ Facade
|
||||
|
||||
- G::File
|
||||
- GNet::Address
|
||||
|
||||
+ Adapter/Mediator
|
||||
|
||||
- GSmtp::ProtocolMessage
|
||||
|
||||
Lakos' Large Scale C++ Software Design patterns (ISBN 0-201-63362-0):
|
||||
|
||||
+ Insulation; fully insulating concrete class (Meyer's Effective C++ Item 34, pimple pattern)
|
||||
|
||||
- G::DirectoryIterator
|
||||
- GNet::Address
|
||||
- GNet::Resolver
|
||||
- GSmtp::ProtocolMessage
|
||||
|
||||
+ Insulation; protocol class
|
||||
|
||||
- GNet::EventHandler
|
||||
- GSmtp::NewMessage
|
||||
- GSmtp::StoredMessage
|
||||
- GSmtp::ProtocolMessage
|
||||
- GSmtp::ServerProtocol::Sender
|
||||
- GSmtp::ClientProtocol::Sender
|
||||
- GSmtp::ClientProtocol::Callback
|
||||
- GSmtp::ProtocolMessage::Callback
|
||||
- GSmtp::Client::ClientCallback
|
||||
|
||||
Meyer's More Effective C++ patterns (ISBN 0-201-63371-X):
|
||||
|
||||
+ Reference counting (Item 29)
|
||||
|
||||
- GSmtp::MessageStore::Iterator
|
||||
|
||||
+ Lazy evaluation (Item 17)
|
||||
|
||||
- GNet::EventHandlerList::list()
|
||||
|
||||
Other patterns:
|
||||
|
||||
+ Finite state machine
|
||||
|
||||
- GSmtp::ServerProtocol
|
||||
|
||||
|
||||
|
||||
|
6
doc/doxygen_header.html
Normal file
6
doc/doxygen_header.html
Normal file
@ -0,0 +1,6 @@
|
||||
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
|
||||
<html><head><meta http-equiv="Content-Type" content="text/html;charset=iso-8859-1">
|
||||
<title>E-MailRelay: $title</title>
|
||||
<meta name="robots" content="noindex,nofollow">
|
||||
<link href="doxygen.css" rel="stylesheet" type="text/css">
|
||||
</head><body bgcolor="#ffffff">
|
@ -34,7 +34,7 @@ emailrelay \- e-mail transfer agent
|
||||
is an simple e-mail message transfer agent. It is intended to be used
|
||||
on stand-alone machines which have a dial-up connection to an ISP.
|
||||
.LP
|
||||
It runs in two modes: a storage deamon
|
||||
It runs in two main modes: a storage deamon
|
||||
.RI ( --as-server )
|
||||
and a forwarding
|
||||
agent
|
||||
@ -43,8 +43,16 @@ The storage daemon is an SMTP server which stores e-mail
|
||||
messages in a local spool directory. The forwarding agent acts as an
|
||||
SMTP client, which passes the spooled e-mail messages on to an ISP's SMTP
|
||||
server.
|
||||
.LP
|
||||
It can also run in a third mode, as a proxy server
|
||||
.RI ( --as-proxy ).
|
||||
In this mode all messages are forwarded immediately to the downstream
|
||||
server.
|
||||
.SH OPTIONS
|
||||
.TP
|
||||
.B \-V,--version
|
||||
Displays version information and exits.
|
||||
.TP
|
||||
.B \-a,--admin \fIadmin-port\fR
|
||||
Enables the administration interface and specifies its listening port number.
|
||||
.TP
|
||||
@ -66,6 +74,9 @@ Records the daemon process-id in the given file.
|
||||
.B \-l,--log
|
||||
Writes log information on standard error (if open) and syslog (if not disabled).
|
||||
.TP
|
||||
.B \-m,--immediate
|
||||
Forwards each message as soon as it is received (requires \fI--forward-to\fR).
|
||||
.TP
|
||||
.B \-n,--no-syslog
|
||||
Disables syslog output.
|
||||
.TP
|
||||
@ -93,8 +104,11 @@ Generates more verbose logging (if compiled-in and logging enabled and stderr op
|
||||
.B \-x,--dont-serve
|
||||
Stops the process acting as a server (usually used with \fI--forward\fR).
|
||||
.TP
|
||||
.B \-V,--version
|
||||
Displays version information and exits.
|
||||
.B \-y,--as-proxy \fIhost:port\fR
|
||||
Equivalent to \fI--close-stderr\fR \fI--log\fR \fI--immediate\fR \fI--forward-to\fR.
|
||||
.TP
|
||||
.B \-z,--filter \fIprogram\fR
|
||||
Defines a mail preprocessor (disallowed if running as root).
|
||||
.SH "DIAGNOSTICS"
|
||||
If the
|
||||
.IR --log ,
|
||||
@ -119,7 +133,7 @@ Failed e-mail messages are kept in the spool directory and given
|
||||
a
|
||||
.I .bad
|
||||
filename suffix. The failure reason is usually recorded within the
|
||||
envelope file iteself.
|
||||
envelope file itself.
|
||||
.SH FILES
|
||||
.IP \(bu 2
|
||||
/usr/local/sbin/emailrelay
|
||||
@ -136,6 +150,8 @@ envelope file iteself.
|
||||
.IP \(bu 2
|
||||
/usr/local/share/emailrelay/emailrelay-deliver.sh
|
||||
.IP \(bu 2
|
||||
/usr/local/share/emailrelay/emailrelay-filter.sh
|
||||
.IP \(bu 2
|
||||
/usr/local/man/man1/emailrelay.1
|
||||
.IP \(bu 2
|
||||
/usr/local/man/man1/emailrelay-poke.1
|
||||
@ -151,6 +167,5 @@ E-MailRelay reference
|
||||
.BR syslog (3),
|
||||
.BR pppd (8),
|
||||
.BR init.d (7)
|
||||
.BR emailrelay-poke (1),
|
||||
.SH AUTHOR
|
||||
Graeme Walker, mailto:graeme_walker@users.sourceforge.net
|
||||
|
BIN
doc/graphics/bullet.gif
Normal file
BIN
doc/graphics/bullet.gif
Normal file
Binary file not shown.
After Width: | Height: | Size: 573 B |
13
doc/index.html
Normal file → Executable file
13
doc/index.html
Normal file → Executable file
@ -4,12 +4,13 @@
|
||||
<body>
|
||||
<h1>E-MailRelay Documentation</h1>
|
||||
<bl>
|
||||
<li><a href=readme.html>Readme</a></li>
|
||||
<li><a href=userguide.html>User guide</a></li>
|
||||
<li><a href=reference.html>Reference manual</a></li>
|
||||
<li><a href=developer.html>Notes for developers</a></li>
|
||||
<li><a href=html/index.html>Source code documentation</a> (generated by <a href=http://www.doxygen.org>doxygen</a>)</li>
|
||||
<li><a href=man.html>Man page</a> (generated by man2html)</li>
|
||||
<li><a href="readme.html">Readme</a></li>
|
||||
<li><a href="userguide.html">User guide</a></li>
|
||||
<li><a href="reference.html">Reference manual</a></li>
|
||||
<li><a href="developer.html">Notes for developers</a></li>
|
||||
<li><a href="html/index.html">Source code documentation</a> (generated by <a href="http://www.doxygen.org">doxygen</a>, if available)</li>
|
||||
<li><a href="man.html">Man page</a> (generated by man2html, if available)</li>
|
||||
<li><a href="http://emailrelay.sourceforge.net">Web site</a></li>
|
||||
</bl>
|
||||
</body>
|
||||
</html>
|
||||
|
@ -1,70 +1,79 @@
|
||||
E-MailRelay Reference Manual
|
||||
============================
|
||||
E-MailRelay Reference
|
||||
=====================
|
||||
|
||||
Document
|
||||
--------
|
||||
This is the E-MailRelay reference guide. It contains material which is supplementary
|
||||
to the user guide.
|
||||
Introduction
|
||||
------------
|
||||
This is the E-MailRelay reference guide. It contains material which is
|
||||
supplementary to the user guide.
|
||||
|
||||
Usage
|
||||
-----
|
||||
Command line usage
|
||||
------------------
|
||||
The "emailrelay" program supports the following command-line usage:
|
||||
|
||||
emailrelay [<switch> [<switch> ...]]
|
||||
|
||||
where <switch> is:
|
||||
|
||||
* --version (-V)
|
||||
# --version (-V)
|
||||
Displays version information and exits.
|
||||
|
||||
* --admin (-a)
|
||||
# --admin (-a)
|
||||
Enables the administration interface and specifies its listening port number.
|
||||
|
||||
* --as-server (-d)
|
||||
# --as-server (-d)
|
||||
Equivalent to "--close-stderr --log".
|
||||
|
||||
* --close-stderr (-e)
|
||||
# --close-stderr (-e)
|
||||
Closes the standard error stream when daemonising.
|
||||
|
||||
* --forward (-f)
|
||||
# --forward (-f)
|
||||
Forwards stored mail on startup (requires --forward-to).
|
||||
|
||||
* --help (-h)
|
||||
# --help (-h)
|
||||
Displays help text and exits.
|
||||
|
||||
* --pid-file (-i)
|
||||
# --pid-file (-i)
|
||||
Records the daemon process-id in the given file.
|
||||
|
||||
* --log (-l)
|
||||
# --log (-l)
|
||||
Writes log information on standard error (if open) and syslog (if not disabled).
|
||||
|
||||
* --no-syslog (-n)
|
||||
# --immediate (-m)
|
||||
Forwards each message as soon as it is received (requires --forward-to).
|
||||
|
||||
# --no-syslog (-n)
|
||||
Disables syslog output.
|
||||
|
||||
* --forward-to (-o)
|
||||
# --forward-to (-o)
|
||||
Specifies the remote smtp server (required by --forward and --admin).
|
||||
|
||||
* --port (-p)
|
||||
# --port (-p)
|
||||
Specifies the smtp listening port number.
|
||||
|
||||
* --as-client (-q)
|
||||
# --as-client (-q)
|
||||
Equivalent to "--no-syslog --no-daemon --log --dont-serve --forward --forward-to".
|
||||
|
||||
* --remote-clients (-r)
|
||||
# --remote-clients (-r)
|
||||
Allows remote clients to connect.
|
||||
|
||||
* --spool-dir (-s)
|
||||
# --spool-dir (-s)
|
||||
Specifies the spool directory (default is "/usr/local/var/spool/emailrelay").
|
||||
|
||||
* --no-daemon (-t)
|
||||
# --no-daemon (-t)
|
||||
Does not detach from the terminal.
|
||||
|
||||
* --verbose (-v)
|
||||
# --verbose (-v)
|
||||
Generates more verbose logging (if compiled-in and logging enabled and stderr open).
|
||||
|
||||
* --dont-serve (-x)
|
||||
# --dont-serve (-x)
|
||||
Stops the process acting as a server (usually used with --forward).
|
||||
|
||||
# --as-proxy (-y)
|
||||
Equivalent to "--close-stderr --log --immediate --forward-to".
|
||||
|
||||
# --filter (-z)
|
||||
Defines a mail pre-processor (disallowed if running as root).
|
||||
|
||||
If no command-line switches are supplied at all then the default
|
||||
behaviour is:
|
||||
* to run as a daemon, detached from the terminal
|
||||
@ -85,6 +94,11 @@ is provided to run the program...
|
||||
The "--as-server" switch makes sure that logging is enabled and that
|
||||
the standard error stream is closed.
|
||||
|
||||
Note that the test for allowing remote clients only takes account of the local
|
||||
machine's canonical IP address: the connection is regarded as local if the
|
||||
remote IP address matches the IP address of the local machine's canonical
|
||||
hostname (ie. not any DNS aliases).
|
||||
|
||||
Message store
|
||||
-------------
|
||||
Mail messages are stored as text files in the configured spool directory. Each
|
||||
@ -113,37 +127,37 @@ and the failure reason is written into the file.
|
||||
|
||||
SMTP issues
|
||||
-----------
|
||||
Local delivery:
|
||||
# Local delivery:
|
||||
|
||||
E-MailRelay will reject all local recipients, with the exception of "postmaster".
|
||||
This is in line with its intended purpose as a simple mail relay, rather than
|
||||
a fully-fledged routing MTA. Any addressee (except "postmaster") without an "at"
|
||||
sign (@) will be rejected at the time the message is submitted by the e-mail
|
||||
front-end.
|
||||
E-MailRelay will reject all local recipients, with the exception of
|
||||
"postmaster". This is in line with its intended purpose as a simple mail
|
||||
relay, rather than a fully-fledged routing MTA. Any addressee (except
|
||||
"postmaster") without an "at" sign (@) will be rejected at the time the
|
||||
message is submitted by the e-mail front-end.
|
||||
|
||||
Delivery of mail to a local "postmaster" is a feature of E-MailRelay which is
|
||||
provided for completeness and for comformance to the SMTP specification. It is
|
||||
only relevant if you are in the habit of sending mail to yourself as "postmaster";
|
||||
mail to "postmaster" from an external source is not processed by E-MailRelay
|
||||
and should be delivered normally.
|
||||
only relevant if you are in the habit of sending mail to yourself as
|
||||
"postmaster"; mail to "postmaster" from an external source is not processed
|
||||
by E-MailRelay and should be delivered normally.
|
||||
|
||||
Note that E-MailRelay daemon does not actually deliver mail to the postmaster
|
||||
mailbox. All it does is create an envelope and content file in the spool directory
|
||||
with a ".local" suffix. Some external system, such as a shell script run
|
||||
from cron calling "procmail", should be used to process the ".local" files.
|
||||
An example script is provided.
|
||||
mailbox. All it does is create an envelope and content file in the spool
|
||||
directory with a ".local" suffix. Some external system, such as a shell
|
||||
script run from cron calling "procmail", should be used to process the
|
||||
".local" files. An example script is provided.
|
||||
|
||||
Timeouts:
|
||||
# Timeouts:
|
||||
|
||||
Client-side timeouts are not implemented. If the ISP server is very slow then
|
||||
in a typical setup the dial-up line will be dropped due to inactivity. This
|
||||
will have the desired effect of aborting the message submission.
|
||||
|
||||
Message loops:
|
||||
# Message loops:
|
||||
|
||||
Message loops are not detected.
|
||||
|
||||
Eight bit messages:
|
||||
# Eight bit messages:
|
||||
|
||||
The 8BITMIME SMTP extension is supported, however no attempt is made to
|
||||
re-encode 8-bit messages into 7-bit messages if the downstream server
|
||||
@ -151,39 +165,76 @@ Eight bit messages:
|
||||
|
||||
Administration interface
|
||||
------------------------
|
||||
If enabled, the server will provide a network interface for performing administration
|
||||
tasks. This is a simple command-line interface which is compatible with telnet.
|
||||
If enabled, the server will provide a network interface for performing
|
||||
administration tasks. This is a simple command-line interface which is
|
||||
compatible with "telnet".
|
||||
|
||||
Currently the only supported command is "flush", which tries to forward spooled mail
|
||||
to the configured dowstream SMTP server. The downstream server address must have been
|
||||
defined on the "emailrelay" command line at start-up using the "--forward-to" switch;
|
||||
it cannot be specified through the administration interface.
|
||||
Currently the only supported command is "flush", which tries to forward spooled
|
||||
mail to the configured dowstream SMTP server. The downstream server address
|
||||
must have been defined on the "emailrelay" command line at start-up using the
|
||||
"--forward-to" switch; it cannot be specified through the administration
|
||||
interface.
|
||||
|
||||
Mail pre-processing
|
||||
-------------------
|
||||
The "--filter" command-line switch allows you to specify a pre-processor program
|
||||
which operates on mail messages as they pass through the E-MailRelay system. The
|
||||
pre-processor program is run as soon as the mail message has been stored in the
|
||||
spool directory, with the full path of the content file specified on the command
|
||||
line.
|
||||
|
||||
This can be combined with the "--immediate" (or "--as-proxy") switch to implement
|
||||
personal mail pre-processing, without replacing or reconfiguring the system's
|
||||
default MTA.
|
||||
|
||||
For example, the following command will start a proxy server on port 10025
|
||||
which pre-processes mail using the specified filter program, and then
|
||||
forwards the mail on to the system's default MTA (on port 25):
|
||||
|
||||
emailrelay --as-proxy localhost:smtp --port 10025 --no-syslog \
|
||||
--filter ${HOME}/.emailrelay/filter \
|
||||
--spool-dir ${HOME}/.emailrelay/spool
|
||||
|
||||
Security issues
|
||||
---------------
|
||||
E-MailRelay runs with a umask of 177. It does not call exec() or system(). It does not
|
||||
change its effective userid. No configuration parameters can be changed through the
|
||||
administrative interface. By default connections to the SMTP and administrative ports
|
||||
will be rejected if they come from remote machines.
|
||||
A major security concern is the use of an external mail pre-processor (using the
|
||||
--filter switch). In this release this feature is simply disabled if the process
|
||||
is running as root (effective userid is zero). The pre-processor will run as the
|
||||
same userid as the E-MailRelay program, but with an almost empty set of
|
||||
environment variables, and no open file descriptors other than
|
||||
"stdin"/"stdout"/"stderr" open onto "/dev/null". The pre-processor filename has
|
||||
to be configured using a full path, so there is no dependence on the current
|
||||
working directory or the PATH variable.
|
||||
|
||||
Some other points are:
|
||||
* The program runs with a "umask" of 177 so files are created with "-rw-------" permissions.
|
||||
* Strings are dynamically allocated, so buffer overflow/truncation issues are avoided.
|
||||
* By default connections to the SMTP and administrative ports will be rejected if they come from remote machines.
|
||||
* No configuration parameters can be changed through the administrative interface.
|
||||
* No exec(), system() or popen() calls are used other than execve() to spawn the mail pre-processor.
|
||||
|
||||
Files
|
||||
-----
|
||||
By default "make install" installs the following files:
|
||||
By default "make install" installs files in the following locations:
|
||||
* /usr/local/sbin/emailrelay
|
||||
* /usr/local/libexec/emailrelay-poke
|
||||
* /usr/local/libexec/emailrelay.sh
|
||||
* /usr/local/var/spool/emailrelay/empty_file
|
||||
* /usr/local/share/emailrelay/emailrelay-notify.sh
|
||||
* /usr/local/share/emailrelay/emailrelay-deliver.sh
|
||||
* /usr/local/share/emailrelay/emailrelay-process.sh
|
||||
* /usr/local/share/emailrelay/*.html
|
||||
|
||||
This directory structure is constrained by the autoconf and GNU standards. Preferred
|
||||
locations would be something like this:
|
||||
This directory structure is constrained by the "autoconf" and GNU standards.
|
||||
Preferred locations for a GNU/Linux distribution would be something like this:
|
||||
* /usr/sbin/emailrelay
|
||||
* /opt/emailrelay/bin/emailrelay-poke
|
||||
* /sbin/init.d/emailrelay.sh
|
||||
* /var/spool/emailrelay/
|
||||
* /opt/emailrelay/examples/emailrelay-notify.sh
|
||||
* /opt/emailrelay/examples/emailrelay-deliver.sh
|
||||
* /opt/emailrelay/examples/emailrelay-process.sh
|
||||
* /usr/share/doc/packages/emailrelay/*.html
|
||||
|
||||
|
||||
|
||||
|
@ -1,47 +1,49 @@
|
||||
E-MailRelay User Guide
|
||||
======================
|
||||
|
||||
Document
|
||||
--------
|
||||
This document is the user guide for E-MailRelay V0.9.1.
|
||||
|
||||
What is it?
|
||||
-----------
|
||||
E-MailRelay is a simple e-mail store-and-forward transfer agent. It's a program
|
||||
which runs in the background and accepts e-mail front-ends (KMail, Mutt, Netscape etc.),
|
||||
stores the messages on the hard disk, and when next connected to the Internet forwards
|
||||
them to a downstream SMTP server for onward delivery.
|
||||
E-MailRelay is a simple store-and-forward e-mail transfer agent. It's a program
|
||||
which runs in the background and accepts e-mail front-ends (KMail, Outlook,
|
||||
Netscape etc.), stores the messages on the hard disk, and when next connected
|
||||
to the Internet forwards them to a downstream SMTP server for onward delivery.
|
||||
|
||||
The E-MailRelay program ("emailrelay") can run in two modes: a storage daemon, or
|
||||
a forwarding agent. As a storage daemon it waits for connections from your
|
||||
The E-MailRelay program ("emailrelay") can run in two main modes: a storage daemon,
|
||||
or a forwarding agent. As a storage daemon it waits for connections from your
|
||||
e-mail front-end and stores the mail which it receives in a spool directory.
|
||||
As a forwarding agent it pulls messages out of the spool directory and passes
|
||||
them on to a remote server -- typically an ISP mail server.
|
||||
|
||||
E-MailRelay uses the Simple Message Transfer Protocol (SMTP). When running as a
|
||||
storage daemon it acts as an SMTP server, and when running as a forwarding agent it
|
||||
acts as an SMTP client.
|
||||
storage daemon it acts as an SMTP server, and when running as a forwarding
|
||||
agent it acts as an SMTP client.
|
||||
|
||||
E-MailRelay runs on GNU/Linux and Windows.
|
||||
The program can also run as a proxy server. In this mode e-mails submitted at the
|
||||
server interface are passed on to the dowstream server immediately, without
|
||||
spooling. This can be useful when combined with mail pre-processing to do things
|
||||
like encryption, message archiving, addition of digital signatures, etc.
|
||||
|
||||
E-MailRelay runs on GNU/Linux, FreeBSD, Solaris and Windows.
|
||||
|
||||
What it's not
|
||||
-------------
|
||||
E-MailRelay does not get involved in processing incoming e-mail messages; it only
|
||||
operates on outgoing messages. Incoming e-mail messages will probably be retrieved
|
||||
from your ISP by your e-mail front-end program, using the POP3 or IMAP protocols.
|
||||
E-MailRelay does not get involved in processing incoming e-mail messages; it
|
||||
only operates on outgoing messages. Incoming e-mail messages will probably be
|
||||
retrieved from your ISP by your e-mail front-end program, using the POP3 or
|
||||
IMAP protocols.
|
||||
|
||||
E-MailRelay is not a routing MTA. It is designed to be used in situations where all
|
||||
outgoing e-mail message go out to the Internet, so it is not an appropriate choice
|
||||
if you send e-mail to other people who use the same machine or to people who are
|
||||
on the same local area network.
|
||||
E-MailRelay is not a routing MTA. It is designed to be used in situations where
|
||||
all outgoing e-mail message go out to the Internet, so it is not an appropriate
|
||||
choice if you send e-mail to other people who use the same machine or to people
|
||||
who are on the same local area network.
|
||||
|
||||
Why use it?
|
||||
-----------
|
||||
The motivation for developing E-MailRelay is that e-mail store-and-forward using
|
||||
SMTP is conceptually a very simple thing, but most popular MTAs are complex.
|
||||
E-MailRelay just stores messages and then forwards them on to your ISP, whereas a
|
||||
fully-featured MTA does clever things with address re-writing, message routing,
|
||||
local delivery, loop detection, fancy DNS lookups etc.
|
||||
The motivation for developing E-MailRelay is that e-mail store-and-forward
|
||||
using SMTP is conceptually a very simple thing, but most popular MTAs are
|
||||
complex. E-MailRelay just stores messages and then forwards them on to your
|
||||
ISP, whereas a fully-featured MTA does clever things with address re-writing,
|
||||
message routing, local delivery, loop detection, fancy DNS lookups etc.
|
||||
|
||||
In particular the configuration of some popular MTAs is notoriously complex
|
||||
and arcane, whereas the only thing the E-MailRelay system needs is the name of
|
||||
@ -51,10 +53,12 @@ With the move away from dial-up Internet connections a simple store-and-forward
|
||||
MTA like E-MailRelay becomes more relevant to mobile computers and PDAs which
|
||||
need to store mail while away from the network.
|
||||
|
||||
The source code for E-MailRelay is well-structured, portable ANSI C++, without any
|
||||
thrid-party library dependencies. It could therefore be the basis for other SMTP
|
||||
projects such as, for example, an anonymising remailer or a protocol bridge between
|
||||
SMTP and proprietary mail transfer protocols used in private e-mail networks.
|
||||
The source code for E-MailRelay is well-structured, portable ANSI C++, without
|
||||
any third-party library dependencies. It could therefore be the basis for other
|
||||
SMTP projects such as, for example, an anonymising remailer or an encryption
|
||||
gateway. E-MailRelay is well suited to gateway applications (sitting between
|
||||
local e-mail clients and a local routing MTA) because of its functional
|
||||
simplicity.
|
||||
|
||||
Running E-MailRelay
|
||||
-------------------
|
||||
@ -62,8 +66,8 @@ To run E-MailRelay as a storage daemon use the command:
|
||||
|
||||
emailrelay --as-server
|
||||
|
||||
To run E-MailRelay as a forwarding agent (once connected to the Internet), use a
|
||||
command like this:
|
||||
To run E-MailRelay as a forwarding agent (once connected to the Internet), use
|
||||
a command like this:
|
||||
|
||||
emailrelay --as-client mail.myisp.net:smtp
|
||||
|
||||
@ -76,37 +80,44 @@ interface. The administration interface is a command-line network service
|
||||
compatible with telnet. The interface must be enabled using the "--admin"
|
||||
command line switch, specifying the listening port number.)
|
||||
|
||||
To run E-MailRelay as a personal proxy server (on port 10025) to the system's
|
||||
default server use a command like this:
|
||||
|
||||
emailrelay --as-proxy localhost:smtp --no-syslog --port 10025 --filter ${HOME}/.emailrelay/mailfilter --spool-dir ${HOME}/.emailrelay/spool
|
||||
|
||||
where 'mailfilter' is a program which does the appropriate mail processing.
|
||||
|
||||
For more information on the command-line options refer to the reference guide
|
||||
or run:
|
||||
|
||||
emailrelay --help
|
||||
|
||||
|
||||
Starting the daemon at boot-time
|
||||
--------------------------------
|
||||
The standard installation of E-MailRelay (using "make install") puts most of the
|
||||
files into the right places, but it does not set things up so that the daemon
|
||||
starts at boot time, or that e-mail gets forwarded automatically when you
|
||||
connect to the Internet. You have to do those bits yourself because of the
|
||||
The standard installation of E-MailRelay (using "make install") puts most of
|
||||
the files into the right places, but it does not set things up so that the
|
||||
daemon starts at boot time, or that e-mail gets forwarded automatically when
|
||||
you connect to the Internet. You have to do those bits yourself because of the
|
||||
differences between the various GNU/Linux distributions.
|
||||
|
||||
Many systems, including the most popular GNU/Linux distributions, use the
|
||||
System-V mechanism for starting daemons at boot time. The directory "/etc/init.d"
|
||||
(or "/sbin/init.d") contains a start/stop script for each daemon process, and
|
||||
then symbolic links in the "rc<N>.d" subdirectories control which scripts are
|
||||
run when entering or leaving a particular run-level (<N>). The links point back
|
||||
into the start/stop script in the parent directory, using a "S" prefix for the
|
||||
starting link, and a "K" prefix for the stopping link. The numeric part of
|
||||
the link name determines the order in which the links are called.
|
||||
System-V mechanism for starting daemons at boot time. The directory
|
||||
"/etc/init.d" (or "/sbin/init.d") contains a start/stop script for each daemon
|
||||
process, and then symbolic links in the "rc<N>.d" subdirectories control which
|
||||
scripts are run when entering or leaving a particular run-level (<N>). The
|
||||
links point back into the start/stop script in the parent directory, using a
|
||||
"S" prefix for the starting link, and a "K" prefix for the stopping link. The
|
||||
numeric part of the link name determines the order in which the links are
|
||||
called.
|
||||
|
||||
Before you start you will need to know where your "init.d" directory can be found
|
||||
and what your default run level is:
|
||||
Before you start you will need to know where your "init.d" directory can be
|
||||
found and what your default run level is:
|
||||
|
||||
$ ls -d /*/init.d
|
||||
$ runlevel | awk '{print $2}'
|
||||
|
||||
Assuming these are "/etc/init.d" and "5" you should (as root) copy the E-MailRelay
|
||||
start/stop script into "/etc/init.d":
|
||||
Assuming these are "/etc/init.d" and "5" you should (as root) copy the
|
||||
E-MailRelay start/stop script into "/etc/init.d":
|
||||
|
||||
$ cp /usr/local/libexec/emailrelay.sh /etc/init.d
|
||||
|
||||
@ -131,27 +142,28 @@ daemons compete for the standard SMTP listening port):
|
||||
|
||||
(There are also KDE and GNOME GUIs which you can use to do the latter steps.)
|
||||
|
||||
Automatic triggering of onward delivery
|
||||
---------------------------------------
|
||||
This section assumes that you are using "pppd" to establish your dial-up Internet
|
||||
connection. (Note that KDE's "kppp" and Red Hat's "rp3" are graphical front-ends
|
||||
to the underlying "pppd" daemon.)
|
||||
Triggering onward delivery
|
||||
--------------------------
|
||||
This section assumes that you are using "pppd" to establish your dial-up
|
||||
Internet connection. (Note that KDE's "kppp" and Red Hat's "rp3" are graphical
|
||||
front-ends to the underlying "pppd" daemon.)
|
||||
|
||||
The ppp daemon calls the script "/etc/ppp/ip-up" when it has successfully established
|
||||
a dial-up link to your ISP. This script will probably set up IP routes, update the
|
||||
DNS configuration, initialise a firewall, run "fetchmail" and "sendmail", etc. It may
|
||||
also call out to another script, "ip-up.local" which is available for you to put
|
||||
stuff into without having to grub around inside "ip-up" itself.
|
||||
The ppp daemon calls the script "/etc/ppp/ip-up" when it has successfully
|
||||
established a dial-up link to your ISP. This script will probably set up IP
|
||||
routes, update the DNS configuration, initialise a firewall, run "fetchmail"
|
||||
and "sendmail", etc. It may also call out to another script, "ip-up.local"
|
||||
which is available for you to put stuff into without having to grub around
|
||||
inside "ip-up" itself.
|
||||
|
||||
The simplest approach for editing "ip-up" is to look for a "sendmail -q" line. If
|
||||
you find "sendmail -q" then it should be sufficient to replace it with this:
|
||||
The simplest approach for editing "ip-up" is to look for a "sendmail -q" line.
|
||||
If you find "sendmail -q" then it should be sufficient to replace it with this:
|
||||
|
||||
emailrelay --as-client <myisp>:smtp
|
||||
|
||||
where you substitute your ISP's SMTP server address for <myisp>.
|
||||
|
||||
Or if your "ip-up" calls out to "ip-up.local" then create a two-line "ip-up.local"
|
||||
script:
|
||||
Or if your "ip-up" calls out to "ip-up.local" then create a two-line
|
||||
"ip-up.local" script:
|
||||
|
||||
$ cd /etc/ppp
|
||||
$ cat << EOF > ip-up.local
|
||||
@ -160,28 +172,53 @@ script:
|
||||
EOF
|
||||
$ chmod +x ip-up.local
|
||||
|
||||
Notification of failed e-mails
|
||||
------------------------------
|
||||
If e-mail messages become corrupted or inaccessible within the spool directory
|
||||
then they will get failed within the E-MailRelay system. In order to get failed
|
||||
e-mails to 'bounce' back into your in-tray you will need to run the
|
||||
"emailrelay-notify.sh" script periodically. Note that this script requires
|
||||
that you have "procmail" installed on your system to act as a "delivery agent".
|
||||
|
||||
There are not many ways in which an e-mail can fail within the E-MailRelay
|
||||
system. If everything is set up correctly then perhaps the most likely case is
|
||||
that you have run out of disk space. If you are not too worried about getting
|
||||
failed mail to bounce, or if you do not have a suitable delivery agent, then
|
||||
a simple check in your ".profile" script for "*.bad" files in the spool
|
||||
directory may be sufficient:
|
||||
|
||||
$ cat <<EOF >> ~/.profile
|
||||
if test -f /usr/local/var/spool/emailrelay/*.envelope.bad ; then echo Failed mail >&2 ; fi
|
||||
EOF
|
||||
|
||||
|
||||
Glossary
|
||||
--------
|
||||
|
||||
ISP = Internet Service Provider. The company your modem calls to connect to the Internet.
|
||||
# ISP
|
||||
Internet Service Provider.
|
||||
|
||||
MTA = Message Transfer Agent. Something which accepts incoming e-mail messages and
|
||||
passes them on either to a local user, or to another MTA. A sophisticated MTA program,
|
||||
which is widely used on the Internet, is "sendmail".
|
||||
# MTA
|
||||
Message Transfer Agent. Something which accepts incoming e-mail messages
|
||||
and passes them on either to a local user, or to another MTA. A sophisticated
|
||||
MTA program, which is widely used on the Internet, is "sendmail".
|
||||
|
||||
SMTP = Simple Message Transfer Protocol. A set of rules which dictate how e-mail messages
|
||||
are passed from one part of the e-mail system to the next. The protocol rules are set
|
||||
out in the document called RFC2821.
|
||||
# SMTP
|
||||
Simple Message Transfer Protocol. A set of rules which dictate how
|
||||
e-mail messages are passed from one part of the e-mail system to the next.
|
||||
The protocol rules are set out in the document "RFC2821".
|
||||
|
||||
POP3 = Post Office Protocol 3. A protocol for fetching incoming e-mail messages from your
|
||||
ISP's mail server. Many e-mail front-ends (Mutt, KMail, Netscape, etc) will fetch messages
|
||||
directly from your ISP using the POP protocol, or you may have a program like "fetchmail"
|
||||
doing it on their behalf.
|
||||
# POP3
|
||||
Post Office Protocol 3. A protocol for fetching incoming e-mail messages.
|
||||
Many e-mail front-ends will fetch messages directly from an ISP using the POP
|
||||
protocol, or a program like "fetchmail" may do it on their behalf.
|
||||
|
||||
IMAP = Internet Message Access Protocol. A newer alternative to POP3.
|
||||
# IMAP
|
||||
Internet Message Access Protocol. A newer alternative to POP3.
|
||||
|
||||
PPP = Point to Point Protocol. A low-level protocol used in dial-up connections to an ISP.
|
||||
Usually implemented by the "pppd" program on GNU/Linux.
|
||||
# PPP
|
||||
Point to Point Protocol. A low-level protocol used in dial-up connections
|
||||
to an ISP. Usually implemented by the "pppd" program on GNU/Linux.
|
||||
|
||||
|
||||
|
||||
|
60
emailrelay.spec
Normal file
60
emailrelay.spec
Normal file
@ -0,0 +1,60 @@
|
||||
Summary: Simple e-mail message transfer agent using SMTP
|
||||
Name: emailrelay
|
||||
Version: 0.9.3
|
||||
Release: 1
|
||||
Copyright: GPL
|
||||
Group: System Environment/Daemons
|
||||
Source: http://emailrelay.sourceforge.net/.../emailrelay-src-0.9.3.tar.gz
|
||||
BuildRoot: /tmp/emailrelay-install
|
||||
|
||||
%description
|
||||
E-MailRelay is a simple SMTP store-and-forward message transfer agent (MTA).
|
||||
It runs as an SMTP server, storing incoming e-mail in a local spool directory,
|
||||
and then forwarding the stored messages to a downstream SMTP server on request.
|
||||
It can also run as a proxy server, forwarding (and optionally pre-processing)
|
||||
incoming e-mail as soon as it is received. It does not do any message routing,
|
||||
other than to a local postmaster. Because of this functional simplicity it is
|
||||
extremely easy to configure, typically only requiring the address of the
|
||||
downstream SMTP server to be put on the command line.
|
||||
|
||||
C++ source code is available for Linux, FreeBSD and Windows. Distribution is
|
||||
under the GNU General Public License.
|
||||
|
||||
%prep
|
||||
%setup
|
||||
|
||||
%build
|
||||
./configure
|
||||
make
|
||||
|
||||
%install
|
||||
make install destdir=$RPM_BUILD_ROOT DESTDIR=$RPM_BUILD_ROOT
|
||||
|
||||
%clean
|
||||
rm -rf $RPM_BUILD_ROOT
|
||||
|
||||
%files
|
||||
|
||||
/usr/local/libexec/emailrelay-poke
|
||||
/usr/local/libexec/emailrelay.sh
|
||||
/usr/local/sbin/emailrelay
|
||||
/usr/local/var/spool/emailrelay/empty_file
|
||||
/usr/local/share/emailrelay/emailrelay-notify.sh
|
||||
/usr/local/share/emailrelay/emailrelay-deliver.sh
|
||||
/usr/local/share/emailrelay/emailrelay-process.sh
|
||||
/usr/local/share/emailrelay/readme.html
|
||||
/usr/local/share/emailrelay/developer.html
|
||||
/usr/local/share/emailrelay/reference.html
|
||||
/usr/local/share/emailrelay/userguide.html
|
||||
/usr/local/share/emailrelay/man.html
|
||||
/usr/local/share/emailrelay/index.html
|
||||
/usr/local/share/emailrelay/graphics/bullet.gif
|
||||
/usr/local/share/emailrelay/html/
|
||||
/usr/local/man/man1/emailrelay.1
|
||||
/usr/local/man/man1/emailrelay-poke.1
|
||||
|
||||
%changelog
|
||||
|
||||
* Mon Sep 24 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
- Initial version.
|
||||
|
@ -46,7 +46,8 @@ std::stringstream::stringstream()
|
||||
}
|
||||
|
||||
inline
|
||||
std::stringstream::stringstream( char * , size_t )
|
||||
std::stringstream::stringstream( char * p , size_t n ) :
|
||||
ostrstream( p , n )
|
||||
{
|
||||
}
|
||||
|
||||
|
@ -23,51 +23,51 @@ EXTRA_DIST=garg_win32.cpp \
|
||||
gdirectory_win32.cpp \
|
||||
gfs_win32.cpp \
|
||||
glogoutput_win32.cpp \
|
||||
gpid_win32.cpp \
|
||||
gprocess_win32.cpp \
|
||||
gfile_win32.cpp \
|
||||
gnumber.cpp
|
||||
INCLUDES = -I$(top_srcdir)/lib/gcc2.95
|
||||
noinst_LIBRARIES = libglib.a
|
||||
libglib_a_SOURCES = garg.cpp \
|
||||
garg_unix.cpp \
|
||||
gdaemon_unix.cpp \
|
||||
gdate.cpp \
|
||||
gdatetime.cpp \
|
||||
gdatetime_unix.cpp \
|
||||
gdirectory.cpp \
|
||||
gdirectory_unix.cpp \
|
||||
gexception.cpp \
|
||||
gfile.cpp \
|
||||
gfile_unix.cpp \
|
||||
gfs_unix.cpp \
|
||||
ggetopt.cpp \
|
||||
glog.cpp \
|
||||
glogoutput.cpp \
|
||||
glogoutput_unix.cpp \
|
||||
gpath.cpp \
|
||||
gpid_unix.cpp \
|
||||
gstr.cpp \
|
||||
gtime.cpp \
|
||||
gdef.h \
|
||||
libglib_a_SOURCES = \
|
||||
garg.cpp \
|
||||
garg.h \
|
||||
gdaemon.h \
|
||||
gdate.h \
|
||||
gdatetime.h \
|
||||
gdirectory.h \
|
||||
gexception.h \
|
||||
gfile.h \
|
||||
gfs.h \
|
||||
ggetopt.h \
|
||||
glog.h \
|
||||
glogoutput.h \
|
||||
gnumber.h \
|
||||
gpath.h \
|
||||
gpid.h \
|
||||
gstr.h \
|
||||
gtime.h \
|
||||
gstrings.h \
|
||||
gdebug.h \
|
||||
garg_unix.cpp \
|
||||
gassert.h \
|
||||
gconvert.h \
|
||||
gmemory.h
|
||||
|
||||
gdaemon.h \
|
||||
gdaemon_unix.cpp \
|
||||
gdate.cpp \
|
||||
gdate.h \
|
||||
gdatetime.cpp \
|
||||
gdatetime.h \
|
||||
gdatetime_unix.cpp \
|
||||
gdebug.h \
|
||||
gdef.h \
|
||||
gdirectory.cpp \
|
||||
gdirectory.h \
|
||||
gdirectory_unix.cpp \
|
||||
gexception.cpp \
|
||||
gexception.h \
|
||||
gfile.cpp \
|
||||
gfile.h \
|
||||
gfile_unix.cpp \
|
||||
gfs.h \
|
||||
gfs_unix.cpp \
|
||||
ggetopt.cpp \
|
||||
ggetopt.h \
|
||||
glog.cpp \
|
||||
glog.h \
|
||||
glogoutput.cpp \
|
||||
glogoutput.h \
|
||||
glogoutput_unix.cpp \
|
||||
gmemory.h \
|
||||
gnumber.h \
|
||||
gpath.cpp \
|
||||
gpath.h \
|
||||
gprocess.h \
|
||||
gprocess_unix.cpp \
|
||||
gstr.cpp \
|
||||
gstr.h \
|
||||
gstrings.h \
|
||||
gtime.cpp \
|
||||
gtime.h
|
||||
|
@ -89,11 +89,11 @@ PACKAGE = @PACKAGE@
|
||||
RANLIB = @RANLIB@
|
||||
VERSION = @VERSION@
|
||||
|
||||
EXTRA_DIST = garg_win32.cpp gdaemon_win32.cpp gdatetime_win32.cpp gdirectory_win32.cpp gfs_win32.cpp glogoutput_win32.cpp gpid_win32.cpp gfile_win32.cpp gnumber.cpp
|
||||
EXTRA_DIST = garg_win32.cpp gdaemon_win32.cpp gdatetime_win32.cpp gdirectory_win32.cpp gfs_win32.cpp glogoutput_win32.cpp gprocess_win32.cpp gfile_win32.cpp gnumber.cpp
|
||||
|
||||
INCLUDES = -I$(top_srcdir)/lib/gcc2.95
|
||||
noinst_LIBRARIES = libglib.a
|
||||
libglib_a_SOURCES = garg.cpp garg_unix.cpp gdaemon_unix.cpp gdate.cpp gdatetime.cpp gdatetime_unix.cpp gdirectory.cpp gdirectory_unix.cpp gexception.cpp gfile.cpp gfile_unix.cpp gfs_unix.cpp ggetopt.cpp glog.cpp glogoutput.cpp glogoutput_unix.cpp gpath.cpp gpid_unix.cpp gstr.cpp gtime.cpp gdef.h garg.h gdaemon.h gdate.h gdatetime.h gdirectory.h gexception.h gfile.h gfs.h ggetopt.h glog.h glogoutput.h gnumber.h gpath.h gpid.h gstr.h gtime.h gstrings.h gdebug.h gassert.h gconvert.h gmemory.h
|
||||
libglib_a_SOURCES = garg.cpp garg.h garg_unix.cpp gassert.h gconvert.h gdaemon.h gdaemon_unix.cpp gdate.cpp gdate.h gdatetime.cpp gdatetime.h gdatetime_unix.cpp gdebug.h gdef.h gdirectory.cpp gdirectory.h gdirectory_unix.cpp gexception.cpp gexception.h gfile.cpp gfile.h gfile_unix.cpp gfs.h gfs_unix.cpp ggetopt.cpp ggetopt.h glog.cpp glog.h glogoutput.cpp glogoutput.h glogoutput_unix.cpp gmemory.h gnumber.h gpath.cpp gpath.h gprocess.h gprocess_unix.cpp gstr.cpp gstr.h gstrings.h gtime.cpp gtime.h
|
||||
|
||||
mkinstalldirs = $(SHELL) $(top_srcdir)/mkinstalldirs
|
||||
CONFIG_HEADER = ../../config.h
|
||||
@ -109,7 +109,7 @@ libglib_a_LIBADD =
|
||||
libglib_a_OBJECTS = garg.o garg_unix.o gdaemon_unix.o gdate.o \
|
||||
gdatetime.o gdatetime_unix.o gdirectory.o gdirectory_unix.o \
|
||||
gexception.o gfile.o gfile_unix.o gfs_unix.o ggetopt.o glog.o \
|
||||
glogoutput.o glogoutput_unix.o gpath.o gpid_unix.o gstr.o gtime.o
|
||||
glogoutput.o glogoutput_unix.o gpath.o gprocess_unix.o gstr.o gtime.o
|
||||
CXXFLAGS = @CXXFLAGS@
|
||||
CXXCOMPILE = $(CXX) $(DEFS) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS)
|
||||
CXXLD = $(CXX)
|
||||
@ -229,7 +229,7 @@ garg_unix.o: garg_unix.cpp gdef.h ../../config.h \
|
||||
gdaemon_unix.o: gdaemon_unix.cpp gdef.h ../../config.h \
|
||||
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
|
||||
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gdaemon.h \
|
||||
gexception.h gpath.h gstrings.h gpid.h
|
||||
gexception.h gpath.h gstrings.h gprocess.h
|
||||
gdate.o: gdate.cpp gdef.h ../../config.h ../../lib/gcc2.95/iostream \
|
||||
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
|
||||
../../lib/gcc2.95/limits gdate.h gdatetime.h gexception.h \
|
||||
@ -283,9 +283,10 @@ gpath.o: gpath.cpp gdef.h ../../config.h ../../lib/gcc2.95/iostream \
|
||||
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
|
||||
../../lib/gcc2.95/limits gpath.h gstrings.h gfs.h gstr.h \
|
||||
gexception.h gdebug.h glogoutput.h glog.h gassert.h
|
||||
gpid_unix.o: gpid_unix.cpp gdef.h ../../config.h \
|
||||
gprocess_unix.o: gprocess_unix.cpp gdef.h ../../config.h \
|
||||
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
|
||||
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gpid.h
|
||||
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gprocess.h \
|
||||
gexception.h gpath.h gstrings.h gfs.h glog.h
|
||||
gstr.o: gstr.cpp gdef.h ../../config.h ../../lib/gcc2.95/iostream \
|
||||
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
|
||||
../../lib/gcc2.95/limits gstr.h gexception.h gstrings.h \
|
||||
|
@ -34,11 +34,13 @@ namespace G
|
||||
} ;
|
||||
|
||||
// Class: G::Arg
|
||||
// Description: A command line parser class.
|
||||
// Description: A class which holds a represention of the
|
||||
// argc/argv command line array. Also does simple command line
|
||||
// parsing, and, under Windows, command line splitting (the
|
||||
// single command line string is split into an argv[] array,
|
||||
// including argv[0]).
|
||||
//
|
||||
// Also does command line splitting for Windows:
|
||||
// the single command line string is split into
|
||||
// an argv[] array, including argv[0].
|
||||
// See also: G::GetOpt
|
||||
//
|
||||
class G::Arg
|
||||
{
|
||||
|
@ -31,7 +31,6 @@
|
||||
#define G_ASSERT( test ) G::LogOutput::assertion( __FILE__ , __LINE__ , test , #test )
|
||||
#else
|
||||
#define G_ASSERT( test )
|
||||
//#define G_ASSERT( test ) G::LogOutput::assertion( 0 , 0 , test , NULL )
|
||||
#endif
|
||||
|
||||
#endif
|
||||
|
@ -40,13 +40,22 @@ namespace G
|
||||
// Deamonisation includes fork()ing, detaching from the
|
||||
// controlling terminal, setting the process umask, etc.
|
||||
// The windows implementation does nothing.
|
||||
// See also: G::Process
|
||||
//
|
||||
class G::Daemon
|
||||
{
|
||||
public:
|
||||
G_EXCEPTION( CannotFork , "cannot fork" ) ;
|
||||
G_EXCEPTION( BadPidFile , "invalid pid file" ) ;
|
||||
enum Who { Parent , Child } ;
|
||||
class PidFile // Used by G::Daemon::detach().
|
||||
{
|
||||
public: explicit PidFile( const Path & pid_file ) ;
|
||||
public: PidFile() ;
|
||||
public: void commit() ;
|
||||
private: Path m_path ;
|
||||
private: bool m_valid ;
|
||||
friend class Daemon ;
|
||||
} ;
|
||||
|
||||
static void detach() ;
|
||||
// Detaches from the parent environment.
|
||||
@ -59,20 +68,23 @@ public:
|
||||
// to a file. The path must be absolute.
|
||||
// Throws BadPidFile on error.
|
||||
|
||||
static void closeFiles( bool keep_stderr = false ) ;
|
||||
// Closes all open file descriptors.
|
||||
|
||||
static void closeStderr() ;
|
||||
// Closes stderr.
|
||||
|
||||
static void setUmask() ;
|
||||
// Sets a tight umask.
|
||||
static void detach( PidFile & pid_file ) ;
|
||||
// An overload which allows for a delayed write
|
||||
// of the new process-id to a file. The path
|
||||
// must be absolute.
|
||||
//
|
||||
// A delayed write is useful for network daemons
|
||||
// which open a listening port. You do not want
|
||||
// a second instance, which will fail on startup,
|
||||
// to overwrite the pid file of the running
|
||||
// server. In this situation call PidFile::commit()
|
||||
// just before entering the event loop.
|
||||
//
|
||||
// Throws BadPidFile on error.
|
||||
|
||||
private:
|
||||
Daemon() ;
|
||||
static Who fork() ;
|
||||
static void setsid() ;
|
||||
static void cd( const std::string & ) ;
|
||||
} ;
|
||||
|
||||
#endif
|
||||
|
@ -23,21 +23,54 @@
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gdaemon.h"
|
||||
#include "gpid.h"
|
||||
#include "gprocess.h"
|
||||
|
||||
namespace
|
||||
{
|
||||
void PidFile__testSyntax( const G::Path & pid_file )
|
||||
{
|
||||
if( pid_file != G::Path() && !pid_file.isAbsolute() )
|
||||
throw G::Daemon::BadPidFile(std::string("must be an absolute path: ")+pid_file.str()) ;
|
||||
}
|
||||
|
||||
void PidFile__testCreation( const G::Path & pid_file )
|
||||
{
|
||||
if( pid_file != G::Path() )
|
||||
{
|
||||
std::ofstream tester( pid_file.str().c_str() ) ;
|
||||
if( !tester.good() )
|
||||
throw G::Daemon::BadPidFile(std::string("cannot create file: ")+pid_file.str()) ;
|
||||
}
|
||||
}
|
||||
|
||||
void PidFile__create( const G::Path & pid_file )
|
||||
{
|
||||
if( pid_file != G::Path() )
|
||||
{
|
||||
std::ofstream file( pid_file.str().c_str() ) ;
|
||||
file << G::Process::Id() << std::endl ;
|
||||
if( !file.good() )
|
||||
throw G::Daemon::BadPidFile(std::string("cannot create file: ")+pid_file.str()) ;
|
||||
}
|
||||
}
|
||||
} ;
|
||||
|
||||
//static
|
||||
void G::Daemon::detach( const Path & pid_file )
|
||||
{
|
||||
if( !pid_file.isAbsolute() )
|
||||
throw BadPidFile(std::string("must be an absolute path: ")+pid_file.str()) ;
|
||||
|
||||
if( !std::ofstream(pid_file.str().c_str()).good() )
|
||||
throw BadPidFile(std::string("cannot create file: ")+pid_file.str()) ;
|
||||
PidFile__testSyntax( pid_file ) ;
|
||||
PidFile__testCreation( pid_file ) ;
|
||||
|
||||
detach() ;
|
||||
|
||||
std::ofstream file( pid_file.str().c_str() ) ;
|
||||
file << Pid() << std::endl ;
|
||||
PidFile__create( pid_file ) ;
|
||||
}
|
||||
|
||||
//static
|
||||
void G::Daemon::detach( PidFile & pid_file )
|
||||
{
|
||||
PidFile__testSyntax( pid_file.m_path ) ;
|
||||
detach() ;
|
||||
}
|
||||
|
||||
//static
|
||||
@ -45,13 +78,13 @@ void G::Daemon::detach()
|
||||
{
|
||||
// see Stevens, ISBN 0-201-563137-7, ch 13.
|
||||
|
||||
if( fork() == Parent )
|
||||
if( Process::fork() == Process::Parent )
|
||||
::_exit( 0 ) ;
|
||||
|
||||
setsid() ;
|
||||
cd( "/" ) ;
|
||||
(void) Process::cd( "/" , Process::NoThrow() ) ;
|
||||
|
||||
if( fork() == Parent )
|
||||
if( Process::fork() == Process::Parent )
|
||||
::_exit( 0 ) ;
|
||||
}
|
||||
|
||||
@ -62,46 +95,23 @@ void G::Daemon::setsid()
|
||||
; // no-op
|
||||
}
|
||||
|
||||
void G::Daemon::cd( const std::string & dir )
|
||||
// ===
|
||||
|
||||
G::Daemon::PidFile::PidFile() :
|
||||
m_valid(false)
|
||||
{
|
||||
if( 0 != ::chdir( dir.c_str() ) )
|
||||
; // ignore it
|
||||
}
|
||||
|
||||
void G::Daemon::setUmask()
|
||||
G::Daemon::PidFile::PidFile( const G::Path & path ) :
|
||||
m_path(path) ,
|
||||
m_valid(true)
|
||||
{
|
||||
// (note that ansi std::ofstream does not support file permissions,
|
||||
// so rely on the umask to keep things secure)
|
||||
mode_t new_mode = 0177 ; // create as -rw-------
|
||||
mode_t old_mode = ::umask( new_mode ) ;
|
||||
}
|
||||
|
||||
G::Daemon::Who G::Daemon::fork()
|
||||
void G::Daemon::PidFile::commit()
|
||||
{
|
||||
pid_t pid = ::fork() ;
|
||||
if( pid < 0 )
|
||||
{
|
||||
throw CannotFork() ;
|
||||
}
|
||||
return pid == 0 ? Child : Parent ;
|
||||
if( m_valid )
|
||||
PidFile__create( m_path ) ;
|
||||
}
|
||||
|
||||
void G::Daemon::closeStderr()
|
||||
{
|
||||
::close( STDERR_FILENO ) ;
|
||||
}
|
||||
|
||||
void G::Daemon::closeFiles( bool keep_stderr )
|
||||
{
|
||||
int n = 256U ;
|
||||
long rc = ::sysconf( _SC_OPEN_MAX ) ;
|
||||
if( rc > 0L )
|
||||
n = static_cast<int>( rc ) ;
|
||||
|
||||
for( int fd = 0 ; fd < n ; fd++ )
|
||||
{
|
||||
if( !keep_stderr || fd != STDERR_FILENO )
|
||||
::close( fd ) ;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -23,26 +23,39 @@
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gdaemon.h"
|
||||
#include "gprocess.h"
|
||||
|
||||
//static
|
||||
void G::Daemon::detach( const Path & )
|
||||
{
|
||||
detach() ;
|
||||
}
|
||||
|
||||
//static
|
||||
void G::Daemon::detach( PidFile & )
|
||||
{
|
||||
detach() ;
|
||||
}
|
||||
|
||||
//static
|
||||
void G::Daemon::detach()
|
||||
{
|
||||
(void) ::FreeConsole() ;
|
||||
}
|
||||
|
||||
void G::Daemon::closeFiles( bool keep_stderr )
|
||||
// ===
|
||||
|
||||
G::Daemon::PidFile::PidFile() :
|
||||
m_valid(false)
|
||||
{
|
||||
}
|
||||
|
||||
void G::Daemon::setUmask()
|
||||
G::Daemon::PidFile::PidFile( const Path & ) :
|
||||
m_valid(false)
|
||||
{
|
||||
}
|
||||
|
||||
void G::Daemon::closeStderr()
|
||||
void G::Daemon::PidFile::commit()
|
||||
{
|
||||
}
|
||||
|
||||
|
@ -26,6 +26,7 @@
|
||||
#include "gdebug.h"
|
||||
#include <sys/types.h>
|
||||
#include <time.h>
|
||||
#include <iomanip>
|
||||
|
||||
//static
|
||||
int G::Date::yearUpperLimit()
|
||||
@ -83,10 +84,17 @@ void G::Date::init( const G::DateTime::BrokenDownTime & tm )
|
||||
m_weekday = sunday ;
|
||||
}
|
||||
|
||||
std::string G::Date::string( Format ) const
|
||||
std::string G::Date::string( Format format ) const
|
||||
{
|
||||
const char * sep = format == yyyy_mm_dd_slash ? "/" : "" ;
|
||||
std::stringstream ss ;
|
||||
if( format == yyyy_mm_dd_slash )
|
||||
ss << m_year << "/" << m_month << "/" << m_day ;
|
||||
else
|
||||
ss
|
||||
<< m_year
|
||||
<< std::setw(2) << std::setfill('0') << m_month
|
||||
<< std::setw(2) << std::setfill('0') << m_day ;
|
||||
return ss.str() ;
|
||||
}
|
||||
|
||||
|
@ -38,6 +38,7 @@ namespace G
|
||||
|
||||
// Class: G::Date
|
||||
// Description: A date (dd/mm/yyyy) class.
|
||||
// See also: Time, DateTime
|
||||
//
|
||||
class G::Date
|
||||
{
|
||||
@ -53,7 +54,7 @@ public:
|
||||
august , september , october , november , december } ;
|
||||
|
||||
enum Format
|
||||
{ yyyy_mm_dd_slash } ;
|
||||
{ yyyy_mm_dd_slash , yyyy_mm_dd } ;
|
||||
|
||||
static int yearUpperLimit() ;
|
||||
// Returns the smallest supported year value.
|
||||
@ -69,10 +70,10 @@ public:
|
||||
// Constructor for the current date
|
||||
// in the local timezone.
|
||||
|
||||
explicit Date( const G::DateTime::BrokenDownTime & tm ) ;
|
||||
Date( const G::DateTime::BrokenDownTime & tm ) ;
|
||||
// Constructor for the specified date.
|
||||
|
||||
explicit Date( G::DateTime::EpochTime t , const LocalTime & ) ;
|
||||
Date( G::DateTime::EpochTime t , const LocalTime & ) ;
|
||||
// Constructor for the date in the local
|
||||
// timezone as at the given epoch time.
|
||||
|
||||
@ -86,7 +87,8 @@ public:
|
||||
// Returns the day of the week.
|
||||
|
||||
std::string weekdayString( bool brief = false ) const ;
|
||||
// Returns a string representation of the day of the week.
|
||||
// Returns an english string representation of
|
||||
// the day of the week.
|
||||
|
||||
int monthday() const ;
|
||||
// Returns the day of the month.
|
||||
@ -98,7 +100,7 @@ public:
|
||||
// Returns the month.
|
||||
|
||||
std::string monthString( bool brief = false ) const ;
|
||||
// Returns the month as a string.
|
||||
// Returns the month as a string (in english).
|
||||
|
||||
int year() const ;
|
||||
// Returns the year.
|
||||
@ -106,16 +108,16 @@ public:
|
||||
std::string yearString() const ;
|
||||
// Returns the year as a string.
|
||||
|
||||
Date &operator++() ;
|
||||
Date & operator++() ;
|
||||
// Increments the date by one day.
|
||||
|
||||
Date &operator--() ;
|
||||
Date & operator--() ;
|
||||
// Decrements the date by one day.
|
||||
|
||||
bool operator==( const Date &rhs ) const ;
|
||||
bool operator==( const Date & rhs ) const ;
|
||||
// Comparison operator.
|
||||
|
||||
bool operator!=( const Date &rhs ) const ;
|
||||
bool operator!=( const Date & rhs ) const ;
|
||||
// Comparison operator.
|
||||
|
||||
private:
|
||||
|
@ -172,7 +172,7 @@ std::string G::GetOpt::usageSummaryPartOne() const
|
||||
bool first = true ;
|
||||
for( SwitchSpecMap::const_iterator p = m_spec_map.begin() ; p != m_spec_map.end() ; ++p )
|
||||
{
|
||||
if( ! (*p).second.valued )
|
||||
if( !(*p).second.valued && !(*p).second.hidden )
|
||||
{
|
||||
if( first )
|
||||
ss << "[-" ;
|
||||
@ -191,6 +191,8 @@ std::string G::GetOpt::usageSummaryPartTwo() const
|
||||
std::stringstream ss ;
|
||||
const char * sep = "" ;
|
||||
for( SwitchSpecMap::const_iterator p = m_spec_map.begin() ; p != m_spec_map.end() ; ++p )
|
||||
{
|
||||
if( !(*p).second.hidden )
|
||||
{
|
||||
ss << sep << "[" ;
|
||||
if( (*p).second.name.length() )
|
||||
@ -210,6 +212,7 @@ std::string G::GetOpt::usageSummaryPartTwo() const
|
||||
ss << "]" ;
|
||||
sep = " " ;
|
||||
}
|
||||
}
|
||||
return ss.str() ;
|
||||
}
|
||||
|
||||
@ -222,6 +225,8 @@ std::string G::GetOpt::usageHelpCore( const std::string & prefix , size_t tab_st
|
||||
{
|
||||
std::string result ;
|
||||
for( SwitchSpecMap::const_iterator p = m_spec_map.begin() ; p != m_spec_map.end() ; ++p )
|
||||
{
|
||||
if( !(*p).second.hidden )
|
||||
{
|
||||
std::string line( prefix ) ;
|
||||
line.append( "-" ) ;
|
||||
@ -253,9 +258,14 @@ std::string G::GetOpt::usageHelpCore( const std::string & prefix , size_t tab_st
|
||||
std::string indent( tab_stop , ' ' ) ;
|
||||
line = G::Str::wrap( line , "" , indent , width ) ;
|
||||
}
|
||||
else
|
||||
{
|
||||
line.append( 1U , '\n' ) ;
|
||||
}
|
||||
|
||||
result.append( line ) ;
|
||||
}
|
||||
}
|
||||
return result ;
|
||||
}
|
||||
|
||||
|
@ -39,6 +39,7 @@ namespace G
|
||||
|
||||
// Class: G::GetOpt
|
||||
// Description: A command line switch parser.
|
||||
// See also: G::Arg
|
||||
//
|
||||
class G::GetOpt
|
||||
{
|
||||
@ -60,6 +61,9 @@ public:
|
||||
// <switch-description>
|
||||
// <value-type> -- 0 is none, and 1 is a string
|
||||
// <value-description>
|
||||
//
|
||||
// If the switch-description field is empty
|
||||
// then the switch is hidden.
|
||||
|
||||
Arg args() const ;
|
||||
// Returns all the non-switch command-line arguments.
|
||||
@ -125,10 +129,12 @@ private:
|
||||
std::string name ;
|
||||
std::string description ;
|
||||
bool valued ;
|
||||
bool hidden ;
|
||||
std::string value_description ;
|
||||
SwitchSpec(char c_,const std::string &name_,const std::string &description_,
|
||||
bool v_,const std::string &vd_) :
|
||||
c(c_) , name(name_) , description(description_) ,
|
||||
hidden(description_.empty()) ,
|
||||
valued(v_) , value_description(vd_) {}
|
||||
} ;
|
||||
typedef std::map<char,SwitchSpec GLessAllocator(char,SwitchSpec) > SwitchSpecMap ;
|
||||
|
@ -113,7 +113,7 @@ namespace G
|
||||
// then warning/error messages should also get raised by some another
|
||||
// independent means.
|
||||
//
|
||||
#define G_LOG_OUTPUT( expr , severity ) { G::Log::stream() << G::Log::Line(__FILE__,__LINE__) << expr << G::Log::end(severity) ; }
|
||||
#define G_LOG_OUTPUT( expr , severity ) { try { G::Log::stream() << G::Log::Line(__FILE__,__LINE__) << expr << G::Log::end(severity) ; } catch(...) {} }
|
||||
#if defined(_DEBUG) && ! defined(G_NO_DEBUG)
|
||||
#define G_DEBUG( expr ) G_LOG_OUTPUT( expr , G::Log::s_Debug )
|
||||
#else
|
||||
|
@ -26,8 +26,6 @@
|
||||
#include <cstdlib> // getenv
|
||||
#include <cstring> // strlen
|
||||
|
||||
namespace std { using ::getenv ; using ::strlen ; } ;
|
||||
|
||||
void G::LogOutput::rawOutput( G::Log::Severity severity , const char *message )
|
||||
{
|
||||
std::cerr << message ;
|
||||
|
@ -27,22 +27,45 @@
|
||||
#include "gdef.h"
|
||||
#include <memory>
|
||||
|
||||
#if HAVE_CONFIG_H
|
||||
#include <config.h>
|
||||
#else
|
||||
#ifdef G_WINDOWS
|
||||
#define HAVE_NONCONST_AUTOPTR 0
|
||||
#else
|
||||
#define HAVE_NONCONST_AUTOPTR 1
|
||||
#endif
|
||||
#endif
|
||||
|
||||
// Template function: operator<<=
|
||||
// Description: A fix for the problem of resetting
|
||||
// an auto_ptr portably. MSVC6.0 does not have a reset
|
||||
// method, and GCC has a non-const assignment
|
||||
// method, and GCC 2.95 has a non-const assignment
|
||||
// operators. This means that the MSVC code and
|
||||
// the gcc code for resetting an auto_ptr are
|
||||
// radically different. This operator hides
|
||||
// the GCC code for resetting auto_ptr<>s has to
|
||||
// be quite different. This operator hides
|
||||
// those differences.
|
||||
//
|
||||
// Usage:
|
||||
/// #include <memory>
|
||||
/// #include "gmemory.h"
|
||||
/// {
|
||||
/// std::auto_ptr<Foo> ptr ;
|
||||
/// for( int i = 0 ; i < 10 ; i++ )
|
||||
/// {
|
||||
/// ptr <<= new Foo ;
|
||||
/// if( ptr->fn() )
|
||||
/// eatFoo( ptr->release() ) ;
|
||||
/// }
|
||||
/// }
|
||||
//
|
||||
template <class T>
|
||||
void operator<<=( std::auto_ptr<T> & ap , T * p )
|
||||
{
|
||||
#ifdef G_WINDOWS
|
||||
ap = std::auto_ptr<T>( p ) ;
|
||||
#else
|
||||
#if HAVE_NONCONST_AUTOPTR
|
||||
ap.reset( p ) ;
|
||||
#else
|
||||
ap = std::auto_ptr<T>( p ) ;
|
||||
#endif
|
||||
}
|
||||
|
||||
@ -52,7 +75,6 @@ void operator<<=( std::auto_ptr<T> & ap , T * p )
|
||||
template <class T>
|
||||
void operator<<=( std::auto_ptr<T> & ap , int null_pointer )
|
||||
{
|
||||
//operator<<=<T>( ap , (T*)(0) ) ;
|
||||
T * p = 0 ;
|
||||
ap <<= p ;
|
||||
}
|
||||
|
129
src/glib/gprocess.h
Normal file
129
src/glib/gprocess.h
Normal file
@ -0,0 +1,129 @@
|
||||
//
|
||||
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
//
|
||||
// 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, write to the Free Software
|
||||
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gprocess.h
|
||||
//
|
||||
|
||||
#ifndef G_PROCESS_H
|
||||
#define G_PROCESS_H
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gexception.h"
|
||||
#include "gpath.h"
|
||||
#include <iostream>
|
||||
#include <sys/types.h>
|
||||
#include <string>
|
||||
|
||||
namespace G
|
||||
{
|
||||
class Process ;
|
||||
} ;
|
||||
|
||||
// Class: G::Process
|
||||
// Description: A static interface for doing things with processes.
|
||||
// See also: G::Daemon
|
||||
//
|
||||
class G::Process
|
||||
{
|
||||
public:
|
||||
G_EXCEPTION( CannotFork , "cannot fork()" ) ;
|
||||
G_EXCEPTION( CannotChangeDirectory , "cannot cd()" ) ;
|
||||
G_EXCEPTION( WaitError , "cannot wait()" ) ;
|
||||
G_EXCEPTION( ChildError , "child process terminated abnormally or stopped" ) ;
|
||||
G_EXCEPTION( InvalidPath , "invalid executable path -- must be absolute" ) ;
|
||||
|
||||
enum Who { Parent , Child } ;
|
||||
class IdImp ;
|
||||
class Id // Process-id class.
|
||||
{
|
||||
public: Id() ;
|
||||
public: ~Id() ;
|
||||
public: Id( const Id & other ) ;
|
||||
public: Id & operator=( const Id & rhs ) ;
|
||||
public: bool operator==( const Id & other ) const ;
|
||||
public: std::string str() const ;
|
||||
private: IdImp * m_imp ;
|
||||
friend class Process ;
|
||||
} ;
|
||||
class NoThrow // An overload discriminator for Process.
|
||||
{} ;
|
||||
|
||||
static void closeFiles( bool keep_stderr = false ) ;
|
||||
// Closes all open file descriptors.
|
||||
|
||||
static void closeStderr() ;
|
||||
// Closes stderr.
|
||||
|
||||
static void setUmask() ;
|
||||
// Sets a tight umask.
|
||||
|
||||
static void cd( const Path & dir ) ;
|
||||
// Changes directory.
|
||||
|
||||
static bool cd( const Path & dir , NoThrow ) ;
|
||||
// Changes directory. Returns false on
|
||||
// error.
|
||||
|
||||
static Who fork() ;
|
||||
// Forks a new process.
|
||||
|
||||
static Who fork( Id & child ) ;
|
||||
// Forks a new process. In the parent process
|
||||
// the child process-id is returned by reference.
|
||||
|
||||
static void exec( const Path & exe , const std::string & arg = std::string() ) ;
|
||||
// Executes a program taking reasonable security
|
||||
// precautions.
|
||||
|
||||
static int wait( const Id & child ) ;
|
||||
// Waits for a child process to terminate.
|
||||
// Returns the exit code. Throws exceptions
|
||||
// on error.
|
||||
|
||||
static int wait( const Id & child , int error_return ) ;
|
||||
// Waits for a child process to terminate.
|
||||
// Returns the exit code, or returns 'error_return'
|
||||
// on error.
|
||||
|
||||
static int spawn( const Path & exe , const std::string & arg , int error_return = 127 ) ;
|
||||
// Runs a command in a child process. Returns the
|
||||
// child process's exit code, or 'error_return' on error.
|
||||
|
||||
static bool privileged() ;
|
||||
// Returns true if this process has enhanced security
|
||||
// privileges.
|
||||
|
||||
private:
|
||||
Process() ;
|
||||
static int errno_() ;
|
||||
static void execCore( const Path & , const std::string & ) ;
|
||||
} ;
|
||||
|
||||
namespace G
|
||||
{
|
||||
inline
|
||||
std::ostream & operator<<( std::ostream & stream , const G::Process::Id & id )
|
||||
{
|
||||
return stream << id.str() ;
|
||||
}
|
||||
} ;
|
||||
|
||||
#endif
|
||||
|
249
src/glib/gprocess_unix.cpp
Normal file
249
src/glib/gprocess_unix.cpp
Normal file
@ -0,0 +1,249 @@
|
||||
//
|
||||
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
//
|
||||
// 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, write to the Free Software
|
||||
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gprocess_unix.cpp
|
||||
//
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gprocess.h"
|
||||
#include "gfs.h"
|
||||
#include "glog.h"
|
||||
#include <errno.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/wait.h>
|
||||
#include <sys/stat.h>
|
||||
#include <fcntl.h> // open()
|
||||
|
||||
class G::Process::IdImp
|
||||
{
|
||||
public:
|
||||
pid_t m_pid ;
|
||||
} ;
|
||||
|
||||
// ===
|
||||
|
||||
//static
|
||||
void G::Process::cd( const Path & dir )
|
||||
{
|
||||
if( ! cd(dir,NoThrow()) )
|
||||
throw CannotChangeDirectory( dir.str() ) ;
|
||||
}
|
||||
|
||||
//static
|
||||
bool G::Process::cd( const Path & dir , NoThrow )
|
||||
{
|
||||
return 0 == ::chdir( dir.str().c_str() ) ;
|
||||
}
|
||||
|
||||
//static
|
||||
void G::Process::setUmask()
|
||||
{
|
||||
mode_t new_mode = 0177 ; // create as -rw-------
|
||||
mode_t old_mode = ::umask( new_mode ) ;
|
||||
}
|
||||
|
||||
//static
|
||||
void G::Process::closeStderr()
|
||||
{
|
||||
::close( STDERR_FILENO ) ;
|
||||
}
|
||||
|
||||
//static
|
||||
void G::Process::closeFiles( bool keep_stderr )
|
||||
{
|
||||
int n = 256U ;
|
||||
long rc = ::sysconf( _SC_OPEN_MAX ) ;
|
||||
if( rc > 0L )
|
||||
n = static_cast<int>( rc ) ;
|
||||
|
||||
for( int fd = 0 ; fd < n ; fd++ )
|
||||
{
|
||||
if( !keep_stderr || fd != STDERR_FILENO )
|
||||
::close( fd ) ;
|
||||
}
|
||||
}
|
||||
|
||||
G::Process::Who G::Process::fork()
|
||||
{
|
||||
Id id ;
|
||||
return fork( id ) ;
|
||||
}
|
||||
|
||||
G::Process::Who G::Process::fork( Id & child_pid )
|
||||
{
|
||||
pid_t rc = ::fork() ;
|
||||
const bool ok = rc != -1 ;
|
||||
if( ok )
|
||||
{
|
||||
if( rc != 0 )
|
||||
child_pid.m_imp->m_pid = rc ;
|
||||
}
|
||||
else
|
||||
{
|
||||
throw CannotFork() ;
|
||||
}
|
||||
return rc == 0 ? Child : Parent ;
|
||||
}
|
||||
|
||||
int G::Process::wait( const Id & child_pid )
|
||||
{
|
||||
int status ;
|
||||
for(;;)
|
||||
{
|
||||
G_DEBUG( "G::Process::wait: waiting" ) ;
|
||||
int rc = ::waitpid( child_pid.m_imp->m_pid , &status , 0 ) ;
|
||||
if( rc == -1 && errno_() == EINTR )
|
||||
{
|
||||
; // signal in parent -- keep waiting
|
||||
}
|
||||
else if( rc == -1 )
|
||||
{
|
||||
int error = errno_() ;
|
||||
throw WaitError( std::stringstream() << "errno=" << error ) ;
|
||||
}
|
||||
else
|
||||
{
|
||||
break ;
|
||||
}
|
||||
}
|
||||
G_DEBUG( "G::Process::wait: done" ) ;
|
||||
|
||||
if( ! WIFEXITED(status) )
|
||||
{
|
||||
// uncaught signal or stopped
|
||||
throw ChildError( std::stringstream() << "status=" << status ) ;
|
||||
}
|
||||
|
||||
const int exit_status = WEXITSTATUS(status) ;
|
||||
return exit_status ;
|
||||
}
|
||||
|
||||
int G::Process::wait( const Id & child_pid , int error_return )
|
||||
{
|
||||
try
|
||||
{
|
||||
return wait( child_pid ) ;
|
||||
}
|
||||
catch(...)
|
||||
{
|
||||
}
|
||||
return error_return ;
|
||||
}
|
||||
|
||||
|
||||
//static
|
||||
int G::Process::errno_()
|
||||
{
|
||||
return errno ; // not ::errno or std::errno for gcc2.95
|
||||
}
|
||||
|
||||
void G::Process::exec( const G::Path & exe , const std::string & arg )
|
||||
{
|
||||
if( exe.isRelative() )
|
||||
throw InvalidPath( exe.str() ) ;
|
||||
|
||||
closeFiles() ;
|
||||
(void) ::open( G::FileSystem::nullDevice() , O_RDONLY ) ; // stdin
|
||||
(void) ::open( G::FileSystem::nullDevice() , O_WRONLY ) ; // stdout
|
||||
(void) ::open( G::FileSystem::nullDevice() , O_WRONLY ) ; // stderr
|
||||
|
||||
// TODO: more security stuff required here -- setuid() etc.
|
||||
|
||||
execCore( exe , arg ) ;
|
||||
}
|
||||
|
||||
void G::Process::execCore( const G::Path & exe , const std::string & arg )
|
||||
{
|
||||
char * argv[3U] ;
|
||||
argv[0U] = const_cast<char*>( exe.pathCstr() ) ;
|
||||
argv[1U] = arg.empty() ? static_cast<char*>(NULL) : const_cast<char*>(arg.c_str()) ;
|
||||
argv[2U] = NULL ;
|
||||
|
||||
// TODO: review the set of environment variables
|
||||
char * env[3U] ;
|
||||
std::string path( "PATH=/usr/bin:/bin" ) ; // no "."
|
||||
std::string ifr( "IFR= \t\n" ) ;
|
||||
env[0U] = const_cast<char*>( path.c_str() ) ;
|
||||
env[1U] = const_cast<char*>( ifr.c_str() ) ;
|
||||
env[2U] = NULL ;
|
||||
|
||||
::execve( exe.str().c_str() , argv , env ) ;
|
||||
|
||||
const int error = errno_() ;
|
||||
G_WARNING( "G::Process::exec: execve() returned: errno=" << error << ": " << exe ) ;
|
||||
}
|
||||
|
||||
int G::Process::spawn( const G::Path & exe , const std::string & arg , int error_return )
|
||||
{
|
||||
Id child_pid ;
|
||||
if( fork(child_pid) == Child )
|
||||
{
|
||||
exec( exe , arg ) ;
|
||||
::_exit( error_return ) ;
|
||||
}
|
||||
else
|
||||
{
|
||||
return wait( child_pid , error_return ) ;
|
||||
}
|
||||
}
|
||||
|
||||
bool G::Process::privileged()
|
||||
{
|
||||
return getuid() == 0 || geteuid() == 0 ;
|
||||
}
|
||||
|
||||
// ===
|
||||
|
||||
G::Process::Id::Id() : m_imp(NULL)
|
||||
{
|
||||
m_imp = new IdImp ;
|
||||
m_imp->m_pid = ::getpid() ;
|
||||
}
|
||||
|
||||
G::Process::Id::~Id()
|
||||
{
|
||||
delete m_imp ;
|
||||
}
|
||||
|
||||
G::Process::Id::Id( const Id & other ) :
|
||||
m_imp(NULL)
|
||||
{
|
||||
m_imp = new IdImp ;
|
||||
m_imp->m_pid = other.m_imp->m_pid ;
|
||||
}
|
||||
|
||||
G::Process::Id & G::Process::Id::operator=( const Id & rhs )
|
||||
{
|
||||
m_imp->m_pid = rhs.m_imp->m_pid ;
|
||||
return *this ;
|
||||
}
|
||||
|
||||
std::string G::Process::Id::str() const
|
||||
{
|
||||
std::stringstream ss ;
|
||||
ss << m_imp->m_pid ;
|
||||
return ss.str() ;
|
||||
}
|
||||
|
||||
bool G::Process::Id::operator==( const Id & rhs ) const
|
||||
{
|
||||
return m_imp->m_pid == rhs.m_imp->m_pid ;
|
||||
}
|
||||
|
149
src/glib/gprocess_win32.cpp
Normal file
149
src/glib/gprocess_win32.cpp
Normal file
@ -0,0 +1,149 @@
|
||||
//
|
||||
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
//
|
||||
// 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, write to the Free Software
|
||||
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gprocess_win32.cpp
|
||||
//
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gprocess.h"
|
||||
#include "glog.h"
|
||||
#include <sys/types.h>
|
||||
#include <sys/stat.h>
|
||||
#include <process.h>
|
||||
#include <direct.h>
|
||||
#include <io.h>
|
||||
|
||||
namespace G
|
||||
{
|
||||
const int STDERR_FILENO = 2 ;
|
||||
const int SC_OPEN_MAX = 256 ; // 32 in limits.h !?
|
||||
} ;
|
||||
|
||||
// ===
|
||||
|
||||
class G::Process::IdImp
|
||||
{
|
||||
public:
|
||||
int m_pid ;
|
||||
} ;
|
||||
|
||||
// ===
|
||||
|
||||
G::Process::Id::Id() : m_imp(NULL)
|
||||
{
|
||||
m_imp = new IdImp ;
|
||||
m_imp->m_pid = ::_getpid() ;
|
||||
}
|
||||
|
||||
G::Process::Id::~Id()
|
||||
{
|
||||
delete m_imp ;
|
||||
}
|
||||
|
||||
G::Process::Id::Id( const Id & other ) :
|
||||
m_imp(NULL)
|
||||
{
|
||||
m_imp = new IdImp ;
|
||||
m_imp->m_pid = other.m_imp->m_pid ;
|
||||
}
|
||||
|
||||
G::Process::Id & G::Process::Id::operator=( const Id & rhs )
|
||||
{
|
||||
m_imp->m_pid = rhs.m_imp->m_pid ;
|
||||
return *this ;
|
||||
}
|
||||
|
||||
std::string G::Process::Id::str() const
|
||||
{
|
||||
std::stringstream ss ;
|
||||
ss << m_imp->m_pid ;
|
||||
return ss.str() ;
|
||||
}
|
||||
|
||||
bool G::Process::Id::operator==( const Id & rhs ) const
|
||||
{
|
||||
return m_imp->m_pid == rhs.m_imp->m_pid ;
|
||||
}
|
||||
|
||||
void G::Process::closeFiles( bool keep_stderr )
|
||||
{
|
||||
const int n = SC_OPEN_MAX ;
|
||||
for( int fd = 0 ; fd < n ; fd++ )
|
||||
{
|
||||
if( !keep_stderr || fd != STDERR_FILENO )
|
||||
::_close( fd ) ;
|
||||
}
|
||||
}
|
||||
|
||||
void G::Process::setUmask()
|
||||
{
|
||||
// _umask() is available but not very useful
|
||||
; // no-op
|
||||
}
|
||||
|
||||
void G::Process::closeStderr()
|
||||
{
|
||||
int fd = STDERR_FILENO ;
|
||||
::_close( fd ) ;
|
||||
}
|
||||
|
||||
void G::Process::cd( const Path & dir )
|
||||
{
|
||||
if( !cd(dir,NoThrow()) )
|
||||
throw CannotChangeDirectory( dir.str() ) ;
|
||||
}
|
||||
|
||||
bool G::Process::cd( const Path & dir , NoThrow )
|
||||
{
|
||||
return 0 == ::_chdir( dir.str().c_str() ) ;
|
||||
}
|
||||
|
||||
int G::Process::spawn( const Path & exe , const std::string & arg ,
|
||||
int error_return )
|
||||
{
|
||||
// open file descriptors are inherited across ::_spawn() --
|
||||
// no fcntl() is available to set close-on-exec -- but see
|
||||
// also ::CreateProcess()
|
||||
|
||||
const char * argv [3U] ;
|
||||
argv[0U] = exe.pathCstr() ;
|
||||
argv[1U] = arg.c_str() ;
|
||||
argv[2U] = NULL ;
|
||||
|
||||
const int mode = _P_WAIT ;
|
||||
::_flushall() ;
|
||||
G_LOG( "G::Process::spawn: " << exe << " " << arg ) ;
|
||||
int rc = ::_spawnv( mode , exe.str().c_str() , argv ) ;
|
||||
G_LOG( "G::Process::spawn: done (" << rc << ")" ) ;
|
||||
return rc < 0 ? error_return : rc ;
|
||||
}
|
||||
|
||||
bool G::Process::privileged()
|
||||
{
|
||||
return false ;
|
||||
}
|
||||
|
||||
// not implemented...
|
||||
// Who G::Process::fork() {}
|
||||
// Who G::Process::fork( Id & child ) {}
|
||||
// void G::Process::exec( const Path & exe , const std::string & arg ) {}
|
||||
// int G::Process::wait( const Id & child ) {}
|
||||
// int G::Process::wait( const Id & child , int error_return ) {}
|
||||
|
@ -38,9 +38,9 @@ namespace GNet
|
||||
|
||||
// Class: GNet::Address
|
||||
//
|
||||
// Description: The Address class encapsulates an
|
||||
// IP transport address. The address is stored
|
||||
// internally as a 'sockaddr_in[6]' structure.
|
||||
// Description: The Address class encapsulates an IP
|
||||
// transport address. The address is stored internally
|
||||
// as a 'sockaddr_in/sockaddr_in6' structure.
|
||||
//
|
||||
// See also: GNet::Resolver
|
||||
//
|
||||
@ -87,7 +87,7 @@ public:
|
||||
// given port number. Throws an exception if
|
||||
// an invalid port number. See also: validPort()
|
||||
|
||||
explicit Address( unsigned int port , Localhost ) ;
|
||||
Address( unsigned int port , Localhost ) ;
|
||||
// Constructor for a local INADDR_LOOPBACK address with
|
||||
// the given port number. Throws an exception if
|
||||
// an invalid port number. See also: validPort()
|
||||
@ -105,7 +105,7 @@ public:
|
||||
|
||||
static Address invalidAddress() ;
|
||||
// Returns an invalid address. Should only be
|
||||
// needed by socket classes.
|
||||
// needed by socket and resolver classes.
|
||||
|
||||
static Address localhost( unsigned int port = 0U ) ;
|
||||
// Returns a localhost ("loopback") address.
|
||||
|
@ -47,7 +47,7 @@ namespace GNet
|
||||
// Class: GNet::ClientResolver
|
||||
// Description: A resolver class which calls ClientImp::resolveCon() when done.
|
||||
//
|
||||
class GNet::ClientResolver : public Resolver
|
||||
class GNet::ClientResolver : public GNet:: Resolver
|
||||
{
|
||||
private:
|
||||
ClientImp & m_client_imp ;
|
||||
@ -73,7 +73,7 @@ GNet::ClientResolver::ClientResolver( ClientImp & imp ) :
|
||||
// Class: GNet::ClientImp
|
||||
// Description: A pimple-pattern implementation class for GClient.
|
||||
//
|
||||
class GNet::ClientImp : public EventHandler
|
||||
class GNet::ClientImp : public GNet:: EventHandler
|
||||
{
|
||||
private:
|
||||
ClientResolver m_resolver ;
|
||||
|
@ -29,9 +29,13 @@
|
||||
|
||||
namespace GNet
|
||||
{
|
||||
typedef ::SOCKET Descriptor ; // SOCKET defined in gnet.h
|
||||
typedef ::SOCKET Descriptor ; // (SOCKET is defined in gnet.h)
|
||||
|
||||
bool Descriptor__valid( Descriptor fd ) ;
|
||||
// Tests whether the given network descriptor is valid.
|
||||
|
||||
Descriptor Descriptor__invalid() ;
|
||||
// Returns an invalid network descriptor.
|
||||
} ;
|
||||
|
||||
#endif
|
||||
|
@ -28,7 +28,7 @@
|
||||
#include "gassert.h"
|
||||
#include "glog.h"
|
||||
|
||||
GNet::EventHandler::EventHandler()
|
||||
GNet::EventHandler::~EventHandler()
|
||||
{
|
||||
}
|
||||
|
||||
|
@ -54,13 +54,12 @@ namespace GNet
|
||||
class GNet::EventHandler
|
||||
{
|
||||
public:
|
||||
EventHandler() ;
|
||||
virtual ~EventHandler() ;
|
||||
virtual void readEvent() /*=0*/ ;
|
||||
virtual void writeEvent() /*=0*/ ;
|
||||
virtual void exceptionEvent() /*=0*/ ;
|
||||
private:
|
||||
EventHandler( const EventHandler & ) ;
|
||||
void operator=( const EventHandler & ) ;
|
||||
void operator=( const EventHandler & ) ; // not implemented
|
||||
} ;
|
||||
|
||||
// Class: GNet::EventSources
|
||||
|
@ -40,7 +40,7 @@ namespace GNet
|
||||
// may be created.
|
||||
// See also: GNet::EventSources
|
||||
//
|
||||
class GNet::EventServer : public Server
|
||||
class GNet::EventServer : public GNet:: Server
|
||||
{
|
||||
public:
|
||||
explicit EventServer( unsigned int listening_port ) ;
|
||||
|
@ -32,19 +32,6 @@ GNet::LineBuffer::LineBuffer( const std::string & eol ) :
|
||||
{
|
||||
}
|
||||
|
||||
//static
|
||||
std::string GNet::LineBuffer::asString( char c )
|
||||
{
|
||||
std::stringstream ss ;
|
||||
if( c == '\n' )
|
||||
ss << "\\n" ;
|
||||
else if( c >= ' ' && c < 0x7f )
|
||||
ss << c ;
|
||||
else
|
||||
ss << "\\" << (unsigned int)c ;
|
||||
return ss.str() ;
|
||||
}
|
||||
|
||||
void GNet::LineBuffer::add( const std::string & segment )
|
||||
{
|
||||
size_t n = segment.size() ;
|
||||
|
@ -70,7 +70,6 @@ private:
|
||||
LineBuffer( const LineBuffer & ) ;
|
||||
void operator=( const LineBuffer & ) ;
|
||||
bool terminated() const ;
|
||||
static std::string asString( char c ) ;
|
||||
|
||||
private:
|
||||
G::Strings m_lines ;
|
||||
|
@ -86,7 +86,7 @@ private:
|
||||
// Class: GNet::HostRequest
|
||||
// Description: A derivation of GNet::Request used for hostname lookup requests.
|
||||
//
|
||||
class GNet::HostRequest : public Request
|
||||
class GNet::HostRequest : public GNet:: Request
|
||||
{
|
||||
public:
|
||||
HostRequest( std::string host_name , HWND hwnd , unsigned msg ) ;
|
||||
@ -101,7 +101,7 @@ private:
|
||||
// Class: GNet::ServiceRequest
|
||||
// Description: A derivation of GNet::Request used for service (port) lookup requests.
|
||||
//
|
||||
class GNet::ServiceRequest : public Request
|
||||
class GNet::ServiceRequest : public GNet:: Request
|
||||
{
|
||||
public:
|
||||
ServiceRequest( std::string service_name , bool udp , HWND hwnd , unsigned msg ) ;
|
||||
|
@ -33,7 +33,7 @@
|
||||
// Class: GNet::ResolverImp
|
||||
// Description: A pimple-pattern implementation class for GNet::Resolver.
|
||||
//
|
||||
class GNet::ResolverImp : public EventHandler
|
||||
class GNet::ResolverImp : public GNet:: EventHandler
|
||||
{
|
||||
public:
|
||||
ResolverImp( Resolver & resolver , unsigned int port ) ;
|
||||
|
@ -37,7 +37,7 @@ namespace GNet
|
||||
// Class: GNet::Select
|
||||
// Description: A event-source class which uses ::select().
|
||||
//
|
||||
class GNet::Select : public EventSources
|
||||
class GNet::Select : public GNet:: EventSources
|
||||
{
|
||||
public:
|
||||
G_EXCEPTION( Error , "select() error" ) ;
|
||||
|
@ -44,7 +44,7 @@ namespace GNet
|
||||
// event-source object (such as GNet::Select) for event handling.
|
||||
// See also: GNet::ServerPeer
|
||||
//
|
||||
class GNet::Server : public EventHandler
|
||||
class GNet::Server : public GNet:: EventHandler
|
||||
{
|
||||
public:
|
||||
G_EXCEPTION( CannotBind , "cannot bind the listening port" ) ;
|
||||
@ -99,7 +99,7 @@ private:
|
||||
// delete themselves when the connection is lost.
|
||||
// See also: GNet::Server, GNet::EventHandler
|
||||
//
|
||||
class GNet::ServerPeer : public EventHandler
|
||||
class GNet::ServerPeer : public GNet:: EventHandler
|
||||
{
|
||||
public:
|
||||
ServerPeer( StreamSocket * , Address ) ;
|
||||
|
@ -236,18 +236,26 @@ private:
|
||||
// (The standard pair<> template cannot be used because gcc's auto_ptr<> has
|
||||
// a non-const copy constructor and assignment operator -- the pair<> op=()
|
||||
// fails to compile because the rhs of the 'first' assignment is const,
|
||||
// not matching any op=() in auto_ptr<>. Note the use of const_cast<>() below.)
|
||||
// not matching any op=() in auto_ptr<>. Note the use of const_cast<>()
|
||||
// in the implementation.)
|
||||
//
|
||||
class GNet::AcceptPair
|
||||
{
|
||||
public:
|
||||
typedef std::auto_ptr<StreamSocket> first_type ;
|
||||
typedef Address second_type ;
|
||||
|
||||
first_type first ;
|
||||
second_type second ;
|
||||
|
||||
AcceptPair( StreamSocket * new_p , Address a ) ;
|
||||
// Constructor.
|
||||
|
||||
AcceptPair( const AcceptPair & other ) ;
|
||||
// Copy constructor.
|
||||
|
||||
AcceptPair & operator=( const AcceptPair & rhs ) ;
|
||||
// Assignment operator.
|
||||
} ;
|
||||
|
||||
// ===
|
||||
@ -255,7 +263,7 @@ public:
|
||||
// Class: GNet::StreamSocket
|
||||
// Description: A derivation of Socket for a stream socket.
|
||||
//
|
||||
class GNet::StreamSocket : public Socket
|
||||
class GNet::StreamSocket : public GNet:: Socket
|
||||
{
|
||||
public:
|
||||
StreamSocket() ;
|
||||
@ -293,7 +301,7 @@ private:
|
||||
// Description: A derivation of Socket for a connectionless
|
||||
// datagram socket.
|
||||
//
|
||||
class GNet::DatagramSocket : public Socket
|
||||
class GNet::DatagramSocket : public GNet:: Socket
|
||||
{
|
||||
public:
|
||||
DatagramSocket();
|
||||
|
@ -61,8 +61,8 @@ public:
|
||||
// Returns false on error.
|
||||
|
||||
bool attach( HWND hwnd , unsigned int msg ) ;
|
||||
// Initialises the WinSock library, passin
|
||||
// it the specified window handle an
|
||||
// Initialises the WinSock library, passing
|
||||
// it the specified window handle and
|
||||
// message number. WinSock events are sent
|
||||
// to that window. Returns false on error.
|
||||
//
|
||||
@ -88,7 +88,7 @@ public:
|
||||
// 'msg' parameter.
|
||||
|
||||
virtual void run() ;
|
||||
// Override from EventSources. Calls GGui::Pump::run()
|
||||
// Override from EventSources. Calls GGui::Pump::run().
|
||||
|
||||
virtual void quit() ;
|
||||
// Override from EventSources. Calls GGui::Pump::quit().
|
||||
|
@ -24,28 +24,45 @@ AM_INSTALL_PROGRAM_FLAGS=-s
|
||||
# change the local-state directory from .../var to .../var/spool/emailrelay
|
||||
localstatedir = ${prefix}/var/spool/emailrelay
|
||||
|
||||
EXTRA_DIST=gmessagestore_win32.cpp emailrelay.dsp empty_file doxygen.cfg
|
||||
EXTRA_DIST=gmessagestore_win32.cpp emailrelay.dsp icon-32.ico empty_file doxygen.cfg emailrelay.rc
|
||||
INCLUDES = -I$(top_srcdir)/lib/gcc2.95 -I$(top_srcdir)/src/glib -I$(top_srcdir)/src/gnet
|
||||
sbin_PROGRAMS = emailrelay
|
||||
libexec_PROGRAMS = emailrelay-poke
|
||||
localstate_DATA = empty_file
|
||||
emailrelay_SOURCES = gadminserver.cpp \
|
||||
gclientprotocol.cpp \
|
||||
gmessagestore.cpp \
|
||||
gmessagestore_unix.cpp \
|
||||
gprotocolmessage.cpp \
|
||||
gserverprotocol.cpp \
|
||||
gsmtpclient.cpp \
|
||||
gsmtpserver.cpp \
|
||||
mailrelay.cpp \
|
||||
emailrelay_SOURCES = \
|
||||
gadminserver.cpp \
|
||||
gadminserver.h \
|
||||
gclientprotocol.cpp \
|
||||
gclientprotocol.h \
|
||||
gfilestore.cpp \
|
||||
gfilestore.h \
|
||||
gmessagestore.cpp \
|
||||
gmessagestore.h \
|
||||
gmessagestore_unix.cpp \
|
||||
gnewfile.cpp \
|
||||
gnewfile.h \
|
||||
gnewmessage.cpp \
|
||||
gnewmessage.h \
|
||||
gprotocolmessage.cpp \
|
||||
gprotocolmessage.h \
|
||||
gprotocolmessageforward.cpp \
|
||||
gprotocolmessageforward.h \
|
||||
gprotocolmessagestore.cpp \
|
||||
gprotocolmessagestore.h \
|
||||
gserverprotocol.cpp \
|
||||
gserverprotocol.h \
|
||||
gsmtp.h \
|
||||
gsmtpclient.cpp \
|
||||
gsmtpclient.h \
|
||||
gsmtpserver.h
|
||||
gsmtpserver.cpp \
|
||||
gsmtpserver.h \
|
||||
gstoredfile.cpp \
|
||||
gstoredfile.h \
|
||||
gstoredmessage.cpp \
|
||||
gstoredmessage.h \
|
||||
gverifier.cpp \
|
||||
gverifier.h \
|
||||
mailrelay.cpp
|
||||
emailrelay_poke_SOURCES=poke.c
|
||||
emailrelay_LDADD = $(top_builddir)/src/glib/libglib.a $(top_builddir)/src/gnet/libgnet.a
|
||||
|
||||
|
@ -95,12 +95,12 @@ AM_INSTALL_PROGRAM_FLAGS = -s
|
||||
# change the local-state directory from .../var to .../var/spool/emailrelay
|
||||
localstatedir = ${prefix}/var/spool/emailrelay
|
||||
|
||||
EXTRA_DIST = gmessagestore_win32.cpp emailrelay.dsp empty_file doxygen.cfg
|
||||
EXTRA_DIST = gmessagestore_win32.cpp emailrelay.dsp icon-32.ico empty_file doxygen.cfg emailrelay.rc
|
||||
INCLUDES = -I$(top_srcdir)/lib/gcc2.95 -I$(top_srcdir)/src/glib -I$(top_srcdir)/src/gnet
|
||||
sbin_PROGRAMS = emailrelay
|
||||
libexec_PROGRAMS = emailrelay-poke
|
||||
localstate_DATA = empty_file
|
||||
emailrelay_SOURCES = gadminserver.cpp gclientprotocol.cpp gmessagestore.cpp gmessagestore_unix.cpp gprotocolmessage.cpp gserverprotocol.cpp gsmtpclient.cpp gsmtpserver.cpp mailrelay.cpp gadminserver.h gclientprotocol.h gmessagestore.h gprotocolmessage.h gserverprotocol.h gsmtp.h gsmtpclient.h gsmtpserver.h
|
||||
emailrelay_SOURCES = gadminserver.cpp gadminserver.h gclientprotocol.cpp gclientprotocol.h gfilestore.cpp gfilestore.h gmessagestore.cpp gmessagestore.h gmessagestore_unix.cpp gnewfile.cpp gnewfile.h gnewmessage.cpp gnewmessage.h gprotocolmessage.cpp gprotocolmessage.h gprotocolmessageforward.cpp gprotocolmessageforward.h gprotocolmessagestore.cpp gprotocolmessagestore.h gserverprotocol.cpp gserverprotocol.h gsmtp.h gsmtpclient.cpp gsmtpclient.h gsmtpserver.cpp gsmtpserver.h gstoredfile.cpp gstoredfile.h gstoredmessage.cpp gstoredmessage.h gverifier.cpp gverifier.h mailrelay.cpp
|
||||
|
||||
emailrelay_poke_SOURCES = poke.c
|
||||
emailrelay_LDADD = $(top_builddir)/src/glib/libglib.a $(top_builddir)/src/gnet/libgnet.a
|
||||
@ -118,9 +118,11 @@ emailrelay_poke_OBJECTS = poke.o
|
||||
emailrelay_poke_LDADD = $(LDADD)
|
||||
emailrelay_poke_DEPENDENCIES =
|
||||
emailrelay_poke_LDFLAGS =
|
||||
emailrelay_OBJECTS = gadminserver.o gclientprotocol.o gmessagestore.o \
|
||||
gmessagestore_unix.o gprotocolmessage.o gserverprotocol.o gsmtpclient.o \
|
||||
gsmtpserver.o mailrelay.o
|
||||
emailrelay_OBJECTS = gadminserver.o gclientprotocol.o gfilestore.o \
|
||||
gmessagestore.o gmessagestore_unix.o gnewfile.o gnewmessage.o \
|
||||
gprotocolmessage.o gprotocolmessageforward.o gprotocolmessagestore.o \
|
||||
gserverprotocol.o gsmtpclient.o gsmtpserver.o gstoredfile.o \
|
||||
gstoredmessage.o gverifier.o mailrelay.o
|
||||
emailrelay_DEPENDENCIES = $(top_builddir)/src/glib/libglib.a \
|
||||
$(top_builddir)/src/gnet/libgnet.a
|
||||
emailrelay_LDFLAGS =
|
||||
@ -306,10 +308,10 @@ gadminserver.o: gadminserver.cpp ../../src/glib/gdef.h ../../config.h \
|
||||
../../src/glib/gexception.h ../../src/gnet/gevent.h \
|
||||
../../src/gnet/gdescriptor.h ../../src/gnet/gselect.h \
|
||||
../../src/gnet/glinebuffer.h ../../src/glib/gstrings.h \
|
||||
gserverprotocol.h gprotocolmessage.h gsmtpclient.h \
|
||||
gserverprotocol.h gprotocolmessage.h gverifier.h gsmtpclient.h \
|
||||
../../src/gnet/gclient.h gclientprotocol.h gmessagestore.h \
|
||||
../../src/glib/gpath.h ../../src/glib/gstr.h \
|
||||
../../src/glib/gmemory.h
|
||||
gnewmessage.h gstoredmessage.h ../../src/glib/gpath.h \
|
||||
../../src/glib/gstr.h ../../src/glib/gmemory.h
|
||||
gclientprotocol.o: gclientprotocol.cpp ../../src/glib/gdef.h \
|
||||
../../config.h ../../lib/gcc2.95/iostream \
|
||||
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
|
||||
@ -319,32 +321,76 @@ gclientprotocol.o: gclientprotocol.cpp ../../src/glib/gdef.h \
|
||||
../../src/glib/gfile.h ../../src/glib/gpath.h \
|
||||
../../src/glib/gstrings.h ../../src/glib/gstr.h \
|
||||
../../src/glib/gmemory.h gclientprotocol.h gmessagestore.h \
|
||||
../../src/gnet/gresolve.h ../../src/glib/gassert.h \
|
||||
gnewmessage.h gstoredmessage.h ../../src/gnet/gresolve.h \
|
||||
../../src/glib/gassert.h ../../src/glib/glogoutput.h
|
||||
gfilestore.o: gfilestore.cpp ../../src/glib/gdef.h ../../config.h \
|
||||
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
|
||||
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gsmtp.h \
|
||||
../../src/gnet/gnet.h ../../src/glib/glog.h gfilestore.h \
|
||||
gmessagestore.h gnewmessage.h gstoredmessage.h \
|
||||
../../src/glib/gstrings.h ../../src/glib/gpath.h \
|
||||
../../src/glib/gexception.h gnewfile.h gstoredfile.h \
|
||||
../../src/glib/gprocess.h ../../src/glib/gdirectory.h \
|
||||
../../src/glib/gmemory.h ../../src/glib/gfile.h \
|
||||
../../src/glib/gstr.h ../../src/glib/gassert.h \
|
||||
../../src/glib/glogoutput.h
|
||||
gmessagestore.o: gmessagestore.cpp ../../src/glib/gdef.h ../../config.h \
|
||||
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
|
||||
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gsmtp.h \
|
||||
../../src/gnet/gnet.h ../../src/glib/glog.h \
|
||||
../../src/glib/gpid.h ../../src/glib/gdirectory.h \
|
||||
../../src/glib/gpath.h ../../src/glib/gstrings.h \
|
||||
gmessagestore.h ../../src/glib/gexception.h \
|
||||
../../src/glib/gmemory.h ../../src/glib/gfile.h \
|
||||
../../src/glib/gstr.h ../../src/glib/gassert.h \
|
||||
../../src/glib/glogoutput.h
|
||||
../../src/gnet/gnet.h ../../src/glib/glog.h gmessagestore.h \
|
||||
gnewmessage.h gstoredmessage.h ../../src/glib/gstrings.h \
|
||||
../../src/glib/gpath.h ../../src/glib/gexception.h \
|
||||
../../src/glib/gassert.h ../../src/glib/glogoutput.h
|
||||
gmessagestore_unix.o: gmessagestore_unix.cpp ../../src/glib/gdef.h \
|
||||
../../config.h ../../lib/gcc2.95/iostream \
|
||||
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
|
||||
../../lib/gcc2.95/limits gsmtp.h ../../src/gnet/gnet.h \
|
||||
../../src/glib/glog.h gmessagestore.h \
|
||||
../../src/glib/gexception.h ../../src/glib/gstrings.h \
|
||||
../../src/glib/gpath.h
|
||||
../../src/glib/glog.h gmessagestore.h gnewmessage.h \
|
||||
gstoredmessage.h ../../src/glib/gstrings.h \
|
||||
../../src/glib/gpath.h ../../src/glib/gexception.h
|
||||
gnewfile.o: gnewfile.cpp ../../src/glib/gdef.h ../../config.h \
|
||||
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
|
||||
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gsmtp.h \
|
||||
../../src/gnet/gnet.h ../../src/glib/glog.h gmessagestore.h \
|
||||
gnewmessage.h gstoredmessage.h ../../src/glib/gstrings.h \
|
||||
../../src/glib/gpath.h ../../src/glib/gexception.h gnewfile.h \
|
||||
gfilestore.h ../../src/glib/gmemory.h ../../src/glib/gprocess.h \
|
||||
../../src/glib/gfile.h ../../src/glib/gassert.h \
|
||||
../../src/glib/glogoutput.h
|
||||
gnewmessage.o: gnewmessage.cpp ../../src/glib/gdef.h ../../config.h \
|
||||
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
|
||||
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gsmtp.h \
|
||||
../../src/gnet/gnet.h ../../src/glib/glog.h gmessagestore.h \
|
||||
gnewmessage.h gstoredmessage.h ../../src/glib/gstrings.h \
|
||||
../../src/glib/gpath.h ../../src/glib/gexception.h
|
||||
gprotocolmessage.o: gprotocolmessage.cpp ../../src/glib/gdef.h \
|
||||
../../config.h ../../lib/gcc2.95/iostream \
|
||||
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
|
||||
../../lib/gcc2.95/limits gsmtp.h ../../src/gnet/gnet.h \
|
||||
../../src/glib/glog.h gprotocolmessage.h \
|
||||
../../src/glib/gstrings.h gmessagestore.h \
|
||||
../../src/glib/gexception.h ../../src/glib/gpath.h \
|
||||
../../src/glib/gstrings.h gverifier.h
|
||||
gprotocolmessageforward.o: gprotocolmessageforward.cpp \
|
||||
../../src/glib/gdef.h ../../config.h ../../lib/gcc2.95/iostream \
|
||||
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
|
||||
../../lib/gcc2.95/limits gsmtp.h ../../src/gnet/gnet.h \
|
||||
../../src/glib/glog.h gprotocolmessageforward.h \
|
||||
gprotocolmessage.h ../../src/glib/gstrings.h gverifier.h \
|
||||
gprotocolmessagestore.h gnewmessage.h gsmtpclient.h \
|
||||
../../src/gnet/glinebuffer.h ../../src/gnet/gclient.h \
|
||||
../../src/gnet/gaddress.h ../../src/glib/gexception.h \
|
||||
../../src/gnet/gsocket.h ../../src/gnet/gevent.h \
|
||||
../../src/gnet/gdescriptor.h gclientprotocol.h gmessagestore.h \
|
||||
gstoredmessage.h ../../src/glib/gpath.h \
|
||||
../../src/glib/gmemory.h ../../src/glib/gstr.h \
|
||||
../../src/glib/gassert.h ../../src/glib/glogoutput.h
|
||||
gprotocolmessagestore.o: gprotocolmessagestore.cpp ../../src/glib/gdef.h \
|
||||
../../config.h ../../lib/gcc2.95/iostream \
|
||||
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
|
||||
../../lib/gcc2.95/limits gsmtp.h ../../src/gnet/gnet.h \
|
||||
../../src/glib/glog.h gprotocolmessagestore.h \
|
||||
gprotocolmessage.h ../../src/glib/gstrings.h gverifier.h \
|
||||
gnewmessage.h gmessagestore.h gstoredmessage.h \
|
||||
../../src/glib/gpath.h ../../src/glib/gexception.h \
|
||||
../../src/glib/gmemory.h ../../src/glib/gstr.h \
|
||||
../../src/glib/gassert.h ../../src/glib/glogoutput.h
|
||||
gserverprotocol.o: gserverprotocol.cpp ../../src/glib/gdef.h \
|
||||
@ -352,7 +398,7 @@ gserverprotocol.o: gserverprotocol.cpp ../../src/glib/gdef.h \
|
||||
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
|
||||
../../lib/gcc2.95/limits gsmtp.h ../../src/gnet/gnet.h \
|
||||
../../src/glib/glog.h gserverprotocol.h gprotocolmessage.h \
|
||||
../../src/glib/gstrings.h ../../src/glib/gdate.h \
|
||||
../../src/glib/gstrings.h gverifier.h ../../src/glib/gdate.h \
|
||||
../../src/glib/gdatetime.h ../../src/glib/gexception.h \
|
||||
../../src/glib/gdebug.h ../../src/glib/glogoutput.h \
|
||||
../../src/glib/gassert.h ../../src/glib/gtime.h \
|
||||
@ -368,7 +414,8 @@ gsmtpclient.o: gsmtpclient.cpp ../../src/glib/gdef.h ../../config.h \
|
||||
../../src/gnet/glinebuffer.h ../../src/gnet/gclient.h \
|
||||
../../src/gnet/gsocket.h ../../src/gnet/gevent.h \
|
||||
../../src/gnet/gdescriptor.h gclientprotocol.h gmessagestore.h \
|
||||
../../src/gnet/gresolve.h
|
||||
gnewmessage.h gstoredmessage.h ../../src/gnet/gresolve.h \
|
||||
../../src/glib/gassert.h ../../src/glib/glogoutput.h
|
||||
gsmtpserver.o: gsmtpserver.cpp ../../src/glib/gdef.h ../../config.h \
|
||||
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
|
||||
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gsmtp.h \
|
||||
@ -377,9 +424,37 @@ gsmtpserver.o: gsmtpserver.cpp ../../src/glib/gdef.h ../../config.h \
|
||||
../../src/gnet/gaddress.h ../../src/glib/gexception.h \
|
||||
../../src/gnet/gevent.h ../../src/gnet/gdescriptor.h \
|
||||
../../src/gnet/gselect.h ../../src/gnet/glinebuffer.h \
|
||||
../../src/glib/gstrings.h gserverprotocol.h gprotocolmessage.h \
|
||||
../../src/gnet/glocal.h ../../src/glib/gdebug.h \
|
||||
../../src/glib/glogoutput.h ../../src/glib/gassert.h
|
||||
../../src/glib/gstrings.h gverifier.h gserverprotocol.h \
|
||||
gprotocolmessage.h gprotocolmessagestore.h gnewmessage.h \
|
||||
gprotocolmessageforward.h gsmtpclient.h \
|
||||
../../src/gnet/gclient.h gclientprotocol.h gmessagestore.h \
|
||||
gstoredmessage.h ../../src/glib/gpath.h \
|
||||
../../src/glib/gmemory.h ../../src/gnet/glocal.h \
|
||||
../../src/glib/gdebug.h ../../src/glib/glogoutput.h \
|
||||
../../src/glib/gassert.h
|
||||
gstoredfile.o: gstoredfile.cpp ../../src/glib/gdef.h ../../config.h \
|
||||
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
|
||||
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gsmtp.h \
|
||||
../../src/gnet/gnet.h ../../src/glib/glog.h gfilestore.h \
|
||||
gmessagestore.h gnewmessage.h gstoredmessage.h \
|
||||
../../src/glib/gstrings.h ../../src/glib/gpath.h \
|
||||
../../src/glib/gexception.h gstoredfile.h \
|
||||
../../src/glib/gmemory.h ../../src/glib/gfile.h \
|
||||
../../src/glib/gstr.h ../../src/glib/gassert.h \
|
||||
../../src/glib/glogoutput.h
|
||||
gstoredmessage.o: gstoredmessage.cpp ../../src/glib/gdef.h \
|
||||
../../config.h ../../lib/gcc2.95/iostream \
|
||||
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
|
||||
../../lib/gcc2.95/limits gsmtp.h ../../src/gnet/gnet.h \
|
||||
../../src/glib/glog.h gstoredmessage.h \
|
||||
../../src/glib/gstrings.h ../../src/glib/gpath.h
|
||||
gverifier.o: gverifier.cpp ../../src/glib/gdef.h ../../config.h \
|
||||
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
|
||||
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gsmtp.h \
|
||||
../../src/gnet/gnet.h ../../src/glib/glog.h gverifier.h \
|
||||
../../src/glib/gstr.h ../../src/glib/gexception.h \
|
||||
../../src/glib/gstrings.h ../../src/glib/gassert.h \
|
||||
../../src/glib/glogoutput.h
|
||||
mailrelay.o: mailrelay.cpp ../../src/glib/gdef.h ../../config.h \
|
||||
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
|
||||
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gsmtp.h \
|
||||
@ -388,12 +463,15 @@ mailrelay.o: mailrelay.cpp ../../src/glib/gdef.h ../../config.h \
|
||||
../../src/gnet/gaddress.h ../../src/glib/gexception.h \
|
||||
../../src/gnet/gevent.h ../../src/gnet/gdescriptor.h \
|
||||
../../src/gnet/gselect.h ../../src/gnet/glinebuffer.h \
|
||||
../../src/glib/gstrings.h gserverprotocol.h gprotocolmessage.h \
|
||||
gsmtpclient.h ../../src/gnet/gclient.h gclientprotocol.h \
|
||||
gmessagestore.h ../../src/glib/gpath.h ../../src/glib/garg.h \
|
||||
../../src/glib/gdaemon.h ../../src/glib/gstr.h gadminserver.h \
|
||||
../../src/glib/ggetopt.h ../../src/glib/gdebug.h \
|
||||
../../src/glib/glogoutput.h ../../src/glib/gassert.h
|
||||
../../src/glib/gstrings.h gverifier.h gserverprotocol.h \
|
||||
gprotocolmessage.h gsmtpclient.h ../../src/gnet/gclient.h \
|
||||
gclientprotocol.h gmessagestore.h gnewmessage.h \
|
||||
gstoredmessage.h ../../src/glib/gpath.h ../../src/glib/garg.h \
|
||||
../../src/glib/gdaemon.h ../../src/glib/gstr.h gfilestore.h \
|
||||
gnewfile.h gadminserver.h ../../src/glib/gprocess.h \
|
||||
../../src/glib/gmemory.h ../../src/glib/ggetopt.h \
|
||||
../../src/glib/gdebug.h ../../src/glib/glogoutput.h \
|
||||
../../src/glib/gassert.h
|
||||
poke.o: poke.c
|
||||
|
||||
info-am:
|
||||
|
@ -3,7 +3,7 @@
|
||||
# General configuration options
|
||||
#---------------------------------------------------------------------------
|
||||
PROJECT_NAME = E-MailRelay
|
||||
PROJECT_NUMBER = 0.9.2
|
||||
PROJECT_NUMBER = 0.9.3
|
||||
OUTPUT_DIRECTORY =
|
||||
OUTPUT_LANGUAGE = English
|
||||
EXTRACT_ALL = YES
|
||||
@ -71,7 +71,7 @@ EXCLUDE_PATTERNS = */*_win32.* */*_ipv6.cpp */old/* */resolverd.cpp
|
||||
EXAMPLE_PATH =
|
||||
EXAMPLE_PATTERNS =
|
||||
IMAGE_PATH =
|
||||
INPUT_FILTER = __TOP_BUILD__/bin/emailrelay-filter.sh
|
||||
INPUT_FILTER = __TOP_BUILD__/bin/emailrelay-doxygen-filter.sh
|
||||
FILTER_SOURCE_FILES = NO
|
||||
#---------------------------------------------------------------------------
|
||||
# configuration options related to the alphabetical class index
|
||||
@ -84,7 +84,7 @@ IGNORE_PREFIX =
|
||||
#---------------------------------------------------------------------------
|
||||
GENERATE_HTML =
|
||||
HTML_OUTPUT = html
|
||||
HTML_HEADER =
|
||||
HTML_HEADER = __TOP_SRC__/doc/doxygen_header.html
|
||||
HTML_FOOTER =
|
||||
HTML_STYLESHEET =
|
||||
HTML_ALIGN_MEMBERS =
|
||||
|
@ -175,6 +175,10 @@ SOURCE=..\glib\gfile_win32.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=.\gfilestore.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=..\..\src\glib\gfs_win32.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
@ -211,6 +215,14 @@ SOURCE=..\..\src\main\gmessagestore_win32.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=.\gnewfile.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=.\gnewmessage.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=..\glib\gnumber.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
@ -219,7 +231,7 @@ SOURCE=..\..\src\glib\gpath.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=..\..\src\glib\gpid_win32.cpp
|
||||
SOURCE=..\..\src\glib\gprocess_win32.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
@ -227,6 +239,14 @@ SOURCE=..\..\src\main\gprotocolmessage.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=.\gprotocolmessageforward.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=.\gprotocolmessagestore.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=..\win32\gpump.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
@ -275,6 +295,14 @@ SOURCE=..\..\src\gnet\gsocket_win32.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=.\gstoredfile.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=.\gstoredmessage.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=..\..\src\glib\gstr.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
@ -283,6 +311,10 @@ SOURCE=..\..\src\glib\gtime.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=.\gverifier.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=..\win32\gwinbase.cpp
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
@ -379,6 +411,10 @@ SOURCE=..\..\src\glib\gfile.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=.\gfilestore.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=..\..\src\glib\gfs.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
@ -415,11 +451,19 @@ SOURCE=..\..\src\gnet\gnet.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=.\gnewfile.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=.\gnewmessage.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=..\..\src\glib\gpath.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=..\..\src\glib\gpid.h
|
||||
SOURCE=..\..\src\glib\gprocess.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
@ -427,6 +471,14 @@ SOURCE=..\..\src\main\gprotocolmessage.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=.\gprotocolmessageforward.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=.\gprotocolmessagestore.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=..\..\src\gnet\grequest.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
@ -463,6 +515,14 @@ SOURCE=..\..\src\gnet\gsocket.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=.\gstoredfile.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=.\gstoredmessage.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=..\..\src\glib\gstr.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
@ -475,12 +535,24 @@ SOURCE=..\..\src\glib\gtime.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=.\gverifier.h
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=..\..\src\gnet\gwinsock.h
|
||||
# End Source File
|
||||
# End Group
|
||||
# Begin Group "Resource Files"
|
||||
|
||||
# PROP Default_Filter "ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe"
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=.\emailrelay.rc
|
||||
# End Source File
|
||||
# Begin Source File
|
||||
|
||||
SOURCE=".\icon-32.ico"
|
||||
# End Source File
|
||||
# End Group
|
||||
# End Target
|
||||
# End Project
|
||||
|
2
src/main/emailrelay.rc
Normal file
2
src/main/emailrelay.rc
Normal file
@ -0,0 +1,2 @@
|
||||
#define IDI_ICON1 101
|
||||
IDI_ICON1 ICON DISCARDABLE "icon-32.ico"
|
@ -44,7 +44,7 @@ GSmtp::AdminPeer::AdminPeer( GNet::StreamSocket * s , GNet::Address a , AdminSer
|
||||
{
|
||||
}
|
||||
|
||||
void GSmtp::AdminPeer::onCompletion( std::string s )
|
||||
void GSmtp::AdminPeer::clientDone( std::string s )
|
||||
{
|
||||
if( s.empty() )
|
||||
send( "OK" ) ;
|
||||
|
@ -43,7 +43,7 @@ namespace GSmtp
|
||||
// Class: GSmtp::AdminClient
|
||||
// Description: A private implementation class.
|
||||
//
|
||||
class GSmtp::AdminClient : public GSmtp::Client
|
||||
class GSmtp::AdminClient : public GSmtp:: Client
|
||||
{
|
||||
public:
|
||||
AdminClient( AdminPeer & admin_peer ) ;
|
||||
@ -55,7 +55,7 @@ public:
|
||||
// Description: A derivation of ServerPeer for the administration interface.
|
||||
// See also: AdminServer
|
||||
//
|
||||
class GSmtp::AdminPeer : public GNet::ServerPeer , public GSmtp::Client::ClientCallback
|
||||
class GSmtp::AdminPeer : public GNet::ServerPeer , public GSmtp:: Client::ClientCallback
|
||||
{
|
||||
public:
|
||||
AdminPeer( GNet::StreamSocket * , GNet::Address , AdminServer & server , const std::string & ) ;
|
||||
@ -66,7 +66,7 @@ private:
|
||||
void operator=( const AdminPeer & ) ;
|
||||
virtual void onDelete() ; // from GNet::ServerPeer
|
||||
virtual void onData( const char * , size_t ) ; // from GNet::ServerPeer
|
||||
virtual void onCompletion( std::string ) ; // from Client::ClientCallback
|
||||
virtual void clientDone( std::string ) ; // from Client::ClientCallback
|
||||
bool processLine( const std::string & line ) ;
|
||||
static bool is( const std::string & , const char * ) ;
|
||||
void flush( const std::string & ) ;
|
||||
|
@ -74,7 +74,7 @@ bool GSmtp::ClientProtocol::done() const
|
||||
return m_state == sEnd ;
|
||||
}
|
||||
|
||||
void GSmtp::ClientProtocol::sendComplete()
|
||||
void GSmtp::ClientProtocol::sendDone()
|
||||
{
|
||||
if( m_state == sData )
|
||||
{
|
||||
@ -199,6 +199,12 @@ void GSmtp::ClientProtocol::applyEvent( const Reply & reply )
|
||||
m_state = sSentData ;
|
||||
send( std::string("DATA") ) ;
|
||||
}
|
||||
else if( m_state == sSentRcpt )
|
||||
{
|
||||
G_WARNING( "GSmtp::ClientProtocol: recipient rejected" ) ;
|
||||
m_state = sEnd ;
|
||||
doCallback( false , reply.text() ) ;
|
||||
}
|
||||
else if( m_state == sSentData && reply.is(Reply::OkForData_354) )
|
||||
{
|
||||
m_state = sData ;
|
||||
@ -214,21 +220,28 @@ void GSmtp::ClientProtocol::applyEvent( const Reply & reply )
|
||||
send( "." , true ) ;
|
||||
}
|
||||
}
|
||||
else if( m_state == sDone && reply.is(Reply::Ok_250) )
|
||||
else if( m_state == sDone )
|
||||
{
|
||||
const bool ok = reply.is(Reply::Ok_250) ;
|
||||
m_state = sEnd ;
|
||||
doCallback( ok , ok ? std::string() : reply.text() ) ;
|
||||
}
|
||||
else
|
||||
{
|
||||
G_WARNING( "GSmtp::ClientProtocol: failure in client protocol: " << static_cast<int>(m_state) ) ;
|
||||
m_state = sEnd ; // (was sReset)
|
||||
send( "RSET" ) ; // for good meausre
|
||||
doCallback( false , std::string("unexpected response: ")+reply.text() ) ;
|
||||
}
|
||||
}
|
||||
|
||||
void GSmtp::ClientProtocol::doCallback( bool ok , const std::string & reason )
|
||||
{
|
||||
if( m_callback )
|
||||
{
|
||||
Callback * cb = m_callback ;
|
||||
m_callback = NULL ;
|
||||
cb->callback( true ) ;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
G_WARNING( "GSmtp::ClientProtocol: protocol error: " << static_cast<int>(m_state) ) ;
|
||||
m_state = sReset ;
|
||||
send( "RSET" ) ;
|
||||
cb->protocolDone( ok , reason ) ;
|
||||
}
|
||||
}
|
||||
|
||||
@ -359,3 +372,15 @@ bool GSmtp::ClientProtocolReply::textContains( std::string key ) const
|
||||
return m_text.find(key) != std::string::npos ;
|
||||
}
|
||||
|
||||
// ===
|
||||
|
||||
GSmtp::ClientProtocol::Sender::~Sender()
|
||||
{
|
||||
}
|
||||
|
||||
// ===
|
||||
|
||||
GSmtp::ClientProtocol::Callback::~Callback()
|
||||
{
|
||||
}
|
||||
|
||||
|
@ -111,22 +111,22 @@ public:
|
||||
//
|
||||
// Returns false if not all of the string
|
||||
// was sent, either due to flow control
|
||||
// or disconnection. After false os returned
|
||||
// the user should call sendComplete() once
|
||||
// or disconnection. After false is returned
|
||||
// the user should call sendDone() once
|
||||
// the full string has been sent.
|
||||
|
||||
private: void operator=( const Sender & ) ;
|
||||
public: virtual ~Sender() {}
|
||||
private: void operator=( const Sender & ) ; // not implemented
|
||||
public: virtual ~Sender() ;
|
||||
} ;
|
||||
|
||||
class Callback // A callback interface used by ClientProtocol.
|
||||
{
|
||||
public: virtual void callback( bool ok ) = 0 ;
|
||||
public: virtual void protocolDone( bool ok , const std::string & reason ) = 0 ;
|
||||
// Called once the protocol has finished with
|
||||
// a given message. See ClientProtocol::start().
|
||||
|
||||
private: void operator=( const Callback & ) ;
|
||||
public: virtual ~Callback() {}
|
||||
private: void operator=( const Callback & ) ; // not implemented
|
||||
public: virtual ~Callback() ;
|
||||
} ;
|
||||
|
||||
ClientProtocol( Sender & sender , const std::string & thishost ) ;
|
||||
@ -142,7 +142,7 @@ public:
|
||||
// signal that the message has been
|
||||
// processed.
|
||||
|
||||
void sendComplete() ;
|
||||
void sendDone() ;
|
||||
// Called when a blocked connection becomes unblocked.
|
||||
// See ClientProtocol::Sender::protocolSend().
|
||||
|
||||
@ -160,6 +160,7 @@ private:
|
||||
static std::string crlf() ;
|
||||
void applyEvent( const Reply & event ) ;
|
||||
static bool parseReply( Reply & , const std::string & , std::string & ) ;
|
||||
void doCallback( bool , const std::string & ) ;
|
||||
|
||||
private:
|
||||
enum State { sStart , sSentEhlo , sSentHelo , sSentMail ,
|
||||
|
226
src/main/gfilestore.cpp
Normal file
226
src/main/gfilestore.cpp
Normal file
@ -0,0 +1,226 @@
|
||||
//
|
||||
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
//
|
||||
// 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, write to the Free Software
|
||||
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gfilestore.cpp
|
||||
//
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gfilestore.h"
|
||||
#include "gnewfile.h"
|
||||
#include "gstoredfile.h"
|
||||
#include "gprocess.h"
|
||||
#include "gdirectory.h"
|
||||
#include "gmemory.h"
|
||||
#include "gpath.h"
|
||||
#include "gfile.h"
|
||||
#include "gstr.h"
|
||||
#include "glog.h"
|
||||
#include "gassert.h"
|
||||
#include <iostream>
|
||||
#include <fstream>
|
||||
|
||||
namespace GSmtp
|
||||
{
|
||||
class FileIterator ;
|
||||
} ;
|
||||
|
||||
// Class: FileIterator
|
||||
// Description: A 'body' class for the MessageStore::Iterator
|
||||
// 'handle'. The handle/body pattern allows us to copy
|
||||
// iterators by value, and therefore return them
|
||||
// from MessageStore::iterator().
|
||||
//
|
||||
class GSmtp::FileIterator : public GSmtp:: MessageStore::IteratorImp
|
||||
{
|
||||
public:
|
||||
explicit FileIterator( const G::Directory & dir ) ;
|
||||
virtual std::auto_ptr<GSmtp::StoredMessage> next() ;
|
||||
|
||||
private:
|
||||
G::DirectoryIterator m_iter ;
|
||||
|
||||
private:
|
||||
FileIterator( const FileIterator & ) ;
|
||||
void operator=( const FileIterator & ) ;
|
||||
} ;
|
||||
|
||||
// ===
|
||||
|
||||
GSmtp::FileIterator::FileIterator( const G::Directory & dir ) :
|
||||
m_iter( dir , "*.envelope" )
|
||||
{
|
||||
}
|
||||
|
||||
std::auto_ptr<GSmtp::StoredMessage> GSmtp::FileIterator::next()
|
||||
{
|
||||
while( !m_iter.error() && m_iter.more() )
|
||||
{
|
||||
std::auto_ptr<StoredFile> m( new StoredFile(m_iter.filePath()) ) ;
|
||||
if( m->lock() )
|
||||
{
|
||||
std::string reason ;
|
||||
const bool check = true ; // check for no-remote-recipients
|
||||
if( m->readEnvelope(reason,check) && m->openContent(reason) )
|
||||
return std::auto_ptr<StoredMessage>( m.release() ) ;
|
||||
|
||||
m->fail( reason ) ;
|
||||
}
|
||||
G_WARNING( "GSmtp::MessageStore: cannot process file: \"" << m_iter.filePath() << "\"" ) ;
|
||||
}
|
||||
return std::auto_ptr<StoredMessage>(NULL) ;
|
||||
}
|
||||
|
||||
// ===
|
||||
|
||||
GSmtp::FileStore::FileStore( const G::Path & dir , bool optimise ) :
|
||||
m_seq(1UL) ,
|
||||
m_dir(dir) ,
|
||||
m_optimise(optimise) ,
|
||||
m_empty(false)
|
||||
{
|
||||
checkPath( dir ) ;
|
||||
}
|
||||
|
||||
//static
|
||||
std::string GSmtp::FileStore::x()
|
||||
{
|
||||
return "X-MailRelay-" ;
|
||||
}
|
||||
|
||||
//static
|
||||
std::string GSmtp::FileStore::format()
|
||||
{
|
||||
return "#2821.2" ;
|
||||
}
|
||||
|
||||
//static
|
||||
void GSmtp::FileStore::checkPath( const G::Path & directory_path )
|
||||
{
|
||||
// (void) G::File::mkdir( directory_path ) ;
|
||||
G::Directory dir_test( directory_path ) ;
|
||||
if( ! dir_test.valid() )
|
||||
{
|
||||
throw InvalidDirectory( directory_path.str() ) ;
|
||||
}
|
||||
if( ! dir_test.valid(true) )
|
||||
{
|
||||
G_WARNING( "GSmtp::MessageStore: "
|
||||
<< "directory not writable: \""
|
||||
<< directory_path << "\"" ) ;
|
||||
}
|
||||
}
|
||||
|
||||
std::auto_ptr<std::ostream> GSmtp::FileStore::stream( const G::Path & path )
|
||||
{
|
||||
std::auto_ptr<std::ostream> ptr(
|
||||
new std::ofstream( path.pathCstr() ,
|
||||
std::ios_base::binary | std::ios_base::out | std::ios_base::trunc ) ) ;
|
||||
return ptr ;
|
||||
}
|
||||
|
||||
G::Path GSmtp::FileStore::contentPath( unsigned long seq ) const
|
||||
{
|
||||
return fullPath( filePrefix(seq) + ".content" ) ;
|
||||
}
|
||||
|
||||
G::Path GSmtp::FileStore::envelopePath( unsigned long seq ) const
|
||||
{
|
||||
return fullPath( filePrefix(seq) + ".envelope" ) ;
|
||||
}
|
||||
|
||||
G::Path GSmtp::FileStore::envelopeWorkingPath( unsigned long seq ) const
|
||||
{
|
||||
return fullPath( filePrefix(seq) + ".envelope.new" ) ;
|
||||
}
|
||||
|
||||
std::string GSmtp::FileStore::filePrefix( unsigned long seq ) const
|
||||
{
|
||||
std::stringstream ss ;
|
||||
ss << "emailrelay." << G::Process::Id() << "." << seq ;
|
||||
return ss.str() ;
|
||||
}
|
||||
|
||||
G::Path GSmtp::FileStore::fullPath( const std::string & filename ) const
|
||||
{
|
||||
G::Path p( m_dir ) ;
|
||||
p.pathAppend( filename ) ;
|
||||
return p ;
|
||||
}
|
||||
|
||||
unsigned long GSmtp::FileStore::newSeq()
|
||||
{
|
||||
return m_seq++ ;
|
||||
}
|
||||
|
||||
bool GSmtp::FileStore::empty() const
|
||||
{
|
||||
if( m_optimise )
|
||||
{
|
||||
if( !m_empty )
|
||||
const_cast<FileStore*>(this)->m_empty = emptyCore() ;
|
||||
return m_empty ;
|
||||
}
|
||||
else
|
||||
{
|
||||
return emptyCore() ;
|
||||
}
|
||||
}
|
||||
|
||||
bool GSmtp::FileStore::emptyCore() const
|
||||
{
|
||||
G::Directory dir( m_dir ) ;
|
||||
G::DirectoryIterator iter( dir , "*.envelope" ) ;
|
||||
const bool no_more = iter.error() || !iter.more() ;
|
||||
return no_more ;
|
||||
}
|
||||
|
||||
GSmtp::MessageStore::Iterator GSmtp::FileStore::iterator()
|
||||
{
|
||||
return MessageStore::Iterator( new FileIterator(G::Directory(m_dir) ) ) ;
|
||||
}
|
||||
|
||||
std::auto_ptr<GSmtp::StoredMessage> GSmtp::FileStore::get( unsigned long id )
|
||||
{
|
||||
G::Path path = envelopePath( id ) ;
|
||||
|
||||
std::auto_ptr<StoredFile> message( new StoredFile(path) ) ;
|
||||
|
||||
if( ! message->lock() )
|
||||
throw GetError( path.str() + ": cannot lock the file" ) ;
|
||||
|
||||
std::string reason ;
|
||||
const bool check = false ; // don't check for no-remote-recipients
|
||||
if( ! message->readEnvelope(reason,check) )
|
||||
throw GetError( path.str() + ": cannot read the envelope: " + reason ) ;
|
||||
|
||||
if( ! message->openContent(reason) )
|
||||
throw GetError( path.str() + ": cannot read the content: " + reason ) ;
|
||||
|
||||
std::auto_ptr<StoredMessage> message_base_ptr( message.release() ) ;
|
||||
return message_base_ptr ;
|
||||
}
|
||||
|
||||
std::auto_ptr<GSmtp::NewMessage> GSmtp::FileStore::newMessage( const std::string & from )
|
||||
{
|
||||
m_empty = false ;
|
||||
return std::auto_ptr<NewMessage>( new NewFile(from,*this) ) ;
|
||||
}
|
||||
|
116
src/main/gfilestore.h
Normal file
116
src/main/gfilestore.h
Normal file
@ -0,0 +1,116 @@
|
||||
//
|
||||
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
//
|
||||
// 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, write to the Free Software
|
||||
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gfilestore.h
|
||||
//
|
||||
|
||||
#ifndef G_SMTP_FILE_STORE_H
|
||||
#define G_SMTP_FILE_STORE_H
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gmessagestore.h"
|
||||
#include "gexception.h"
|
||||
#include "gpath.h"
|
||||
#include <memory>
|
||||
#include <string>
|
||||
|
||||
namespace GSmtp
|
||||
{
|
||||
class FileStore ;
|
||||
} ;
|
||||
|
||||
// Class: GSmtp::FileStore
|
||||
// Description: A concrete implementation of the MessageStore
|
||||
// interface, dealing in flat files. Passes out unique sequence
|
||||
// numbers, filesystem paths and i/o streams to NewMessageImp.
|
||||
//
|
||||
class GSmtp::FileStore : public GSmtp:: MessageStore
|
||||
{
|
||||
public:
|
||||
G_EXCEPTION( InvalidDirectory , "invalid spool directory" ) ;
|
||||
G_EXCEPTION( GetError , "error reading specific message" ) ;
|
||||
|
||||
explicit FileStore( const G::Path & dir , bool optimise = false ) ;
|
||||
// Constructor. Throws an exception if the
|
||||
// storage directory is invalid.
|
||||
//
|
||||
// If the optimise flag is set then the implementation
|
||||
// of empty() will be efficient for an empty file
|
||||
// store (ignoring failed and local-delivery messages).
|
||||
// This might be useful for applications in which the
|
||||
// main event loop is used to check for pending jobs.
|
||||
// The disadvantage is that this process will not be
|
||||
// sensititive to messages deposited into its spool
|
||||
// directory by other processes.
|
||||
|
||||
unsigned long newSeq() ;
|
||||
// Hands out a new sequence number.
|
||||
|
||||
std::auto_ptr<std::ostream> stream( const G::Path & path );
|
||||
// Returns a stream to the given content.
|
||||
|
||||
G::Path contentPath( unsigned long seq ) const ;
|
||||
// Returns the path for a content file.
|
||||
|
||||
G::Path envelopePath( unsigned long seq ) const ;
|
||||
// Returns the path for an envelope file.
|
||||
|
||||
G::Path envelopeWorkingPath( unsigned long seq ) const ;
|
||||
// Returns the path for an envelope file
|
||||
// which is in the process of being written.
|
||||
|
||||
virtual bool empty() const ;
|
||||
// Returns true if there are no stored messages.
|
||||
|
||||
virtual std::auto_ptr<StoredMessage> get( unsigned long id ) ;
|
||||
// Extracts a stored message.
|
||||
|
||||
virtual MessageStore::Iterator iterator() ;
|
||||
// Returns an iterator for stored messages.
|
||||
|
||||
virtual std::auto_ptr<NewMessage> newMessage( const std::string & from ) ;
|
||||
// Creates a new message in the store.
|
||||
|
||||
static std::string x() ;
|
||||
// Returns the prefix for envelope header lines.
|
||||
|
||||
static std::string format() ;
|
||||
// Returns an identifier for the storage format
|
||||
// implemented by this class.
|
||||
|
||||
private:
|
||||
static void checkPath( const G::Path & dir ) ;
|
||||
G::Path fullPath( const std::string & filename ) const ;
|
||||
std::string filePrefix( unsigned long seq ) const ;
|
||||
std::string getline( std::istream & ) const ;
|
||||
std::string value( const std::string & ) const ;
|
||||
std::string crlf() const ;
|
||||
bool emptyCore() const ;
|
||||
|
||||
private:
|
||||
unsigned long m_seq ;
|
||||
G::Path m_dir ;
|
||||
bool m_optimise ;
|
||||
bool m_empty ; // mutable
|
||||
} ;
|
||||
|
||||
#endif
|
||||
|
@ -23,178 +23,40 @@
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gpid.h"
|
||||
#include "gdirectory.h"
|
||||
#include "gmessagestore.h"
|
||||
#include "gmemory.h"
|
||||
#include "gpath.h"
|
||||
#include "gfile.h"
|
||||
#include "gstr.h"
|
||||
#include "glog.h"
|
||||
#include "gassert.h"
|
||||
#include <iostream>
|
||||
#include <fstream>
|
||||
|
||||
// Class: MessageStoreImp
|
||||
// Description: Pimple-pattern implementation for MessageStore.
|
||||
// Passes out unique sequence numbers, filesystem paths and
|
||||
// i/o streams to NewMessageImp.
|
||||
//
|
||||
class GSmtp::MessageStoreImp
|
||||
GSmtp::MessageStore * GSmtp::MessageStore::m_this = NULL ;
|
||||
|
||||
GSmtp::MessageStore::MessageStore()
|
||||
{
|
||||
private:
|
||||
G::Path m_dir ;
|
||||
unsigned long m_seq ;
|
||||
public:
|
||||
MessageStoreImp( const G::Path & dir ) ;
|
||||
if( m_this == NULL )
|
||||
m_this = this ;
|
||||
}
|
||||
|
||||
// new message...
|
||||
static void checkPath( const G::Path & ) ;
|
||||
const G::Path & dir() const ;
|
||||
unsigned long newSeq() ;
|
||||
std::auto_ptr<std::ostream> stream( const G::Path & path );
|
||||
G::Path contentPath( unsigned long seq ) const ;
|
||||
G::Path envelopePath( unsigned long seq ) const ;
|
||||
G::Path envelopeWorkingPath( unsigned long seq ) const ;
|
||||
GSmtp::MessageStore & GSmtp::MessageStore::instance()
|
||||
{
|
||||
if( m_this == NULL )
|
||||
throw NoInstance() ;
|
||||
return * m_this ;
|
||||
}
|
||||
|
||||
// stored message...
|
||||
bool empty() const ;
|
||||
std::auto_ptr<StoredMessage> get() ;
|
||||
MessageStore::Iterator iterator() ;
|
||||
|
||||
// both...
|
||||
static std::string x() ;
|
||||
static std::string format() ;
|
||||
private:
|
||||
G::Path fullPath( const std::string & filename ) const ;
|
||||
std::string filePrefix( unsigned long seq ) const ;
|
||||
std::string getline( std::istream & ) const ;
|
||||
std::string value( const std::string & ) const ;
|
||||
std::string crlf() const ;
|
||||
} ;
|
||||
GSmtp::MessageStore::~MessageStore()
|
||||
{
|
||||
if( m_this == this )
|
||||
m_this = NULL ;
|
||||
}
|
||||
|
||||
// ===
|
||||
|
||||
// Class: StoredMessageImp
|
||||
// Description: A concete derived class implementing the
|
||||
// StoredMessage (parent) and Callback (grandparent) interfaces.
|
||||
//
|
||||
class GSmtp::StoredMessageImp : public GSmtp::StoredMessage
|
||||
{
|
||||
public:
|
||||
G_EXCEPTION( InvalidFormat , "invalid format field in envelope" ) ;
|
||||
G_EXCEPTION( NoEnd , "invalid envelope file: no end marker" ) ;
|
||||
G_EXCEPTION( InvalidTo , "invalid 'to' line in envelope file" ) ;
|
||||
G_EXCEPTION( NoRecipients , "no remote recipients" ) ;
|
||||
G_EXCEPTION( OpenError , "cannot open the envelope" ) ;
|
||||
G_EXCEPTION( StreamError , "envelope reading/parsing error" ) ;
|
||||
explicit StoredMessageImp( const G::Path & envelope_path ) ;
|
||||
virtual ~StoredMessageImp() ;
|
||||
bool lock() ;
|
||||
bool readEnvelope( std::string & reason ) ;
|
||||
bool readEnvelopeCore( std::string & reason ) ;
|
||||
bool openContent( std::string & reason ) ;
|
||||
virtual bool eightBit() const ;
|
||||
virtual const std::string & from() const ;
|
||||
virtual const G::Strings & to() const ;
|
||||
virtual void destroy() ;
|
||||
virtual void fail( const std::string & reason ) ;
|
||||
virtual std::auto_ptr<std::istream> extractContentStream() ;
|
||||
|
||||
private:
|
||||
StoredMessageImp( const StoredMessageImp & ) ;
|
||||
void operator=( const StoredMessageImp & ) ;
|
||||
std::string crlf() const ;
|
||||
std::string getline( std::istream & stream ) const ;
|
||||
std::string value( const std::string & s ) const ;
|
||||
G::Path contentPath() const ;
|
||||
|
||||
private:
|
||||
G::Strings m_to_local ;
|
||||
G::Strings m_to_remote ;
|
||||
std::string m_from ;
|
||||
G::Path m_envelope_path ;
|
||||
std::auto_ptr<std::istream> m_content ;
|
||||
bool m_eight_bit ;
|
||||
} ;
|
||||
|
||||
// ===
|
||||
|
||||
// Class: NewMessageImp
|
||||
// Description: A concrete derived class implementing the
|
||||
// NewMessage interface. Writes itself to the i/o streams
|
||||
// supplied by MessageStoreImp.
|
||||
//
|
||||
class GSmtp::NewMessageImp : public GSmtp::NewMessage
|
||||
{
|
||||
public:
|
||||
NewMessageImp( const std::string & from , MessageStoreImp & store ) ;
|
||||
virtual ~NewMessageImp() ;
|
||||
virtual void addTo( const std::string & to , bool local ) ;
|
||||
virtual void addText( const std::string & line ) ;
|
||||
virtual void store() ;
|
||||
private:
|
||||
MessageStoreImp & m_store ;
|
||||
unsigned long m_seq ;
|
||||
std::string m_from ;
|
||||
G::Strings m_to_local ;
|
||||
G::Strings m_to_remote ;
|
||||
std::auto_ptr<std::ostream> m_content ;
|
||||
G::Path m_content_path ;
|
||||
bool m_eight_bit ;
|
||||
private:
|
||||
bool saveEnvelope( std::ostream & stream , const std::string & where ) const ;
|
||||
std::string crlf() const ;
|
||||
static bool isEightBit( const std::string & line ) ;
|
||||
void deliver( const G::Strings & , const G::Path & , const G::Path & , const G::Path & ) ;
|
||||
} ;
|
||||
|
||||
// ===
|
||||
|
||||
// Class: MessageStoreIteratorImp
|
||||
// Description: A 'body' class for the MessageStoreIterator
|
||||
// 'handle'. The handle/body pattern allows us to copy
|
||||
// iterators by value, and therefore return them
|
||||
// from MessageStore::iterator().
|
||||
//
|
||||
class GSmtp::MessageStoreIteratorImp
|
||||
{
|
||||
public:
|
||||
explicit MessageStoreIteratorImp( const G::Directory & dir ) ;
|
||||
std::auto_ptr<GSmtp::StoredMessage> next() ;
|
||||
public:
|
||||
unsigned long m_ref_count ;
|
||||
private:
|
||||
G::DirectoryIterator m_iter ;
|
||||
private:
|
||||
MessageStoreIteratorImp( const MessageStoreIteratorImp & ) ;
|
||||
void operator=( const MessageStoreIteratorImp & ) ;
|
||||
} ;
|
||||
|
||||
// ===
|
||||
|
||||
GSmtp::MessageStoreIteratorImp::MessageStoreIteratorImp( const G::Directory & dir ) :
|
||||
m_iter( dir , "*.envelope" ) ,
|
||||
GSmtp::MessageStore::IteratorImp::IteratorImp() :
|
||||
m_ref_count(1UL)
|
||||
{
|
||||
}
|
||||
|
||||
std::auto_ptr<GSmtp::StoredMessage> GSmtp::MessageStoreIteratorImp::next()
|
||||
GSmtp::MessageStore::IteratorImp::~IteratorImp()
|
||||
{
|
||||
while( !m_iter.error() && m_iter.more() )
|
||||
{
|
||||
std::auto_ptr<StoredMessageImp> m( new StoredMessageImp(m_iter.filePath()) ) ;
|
||||
if( m->lock() )
|
||||
{
|
||||
std::string reason ;
|
||||
if( m->readEnvelope(reason) && m->openContent(reason) )
|
||||
return std::auto_ptr<StoredMessage>( m.release() ) ;
|
||||
|
||||
m->fail( reason ) ;
|
||||
}
|
||||
G_WARNING( "GSmtp::MessageStore: cannot process file: \"" << m_iter.filePath() << "\"" ) ;
|
||||
}
|
||||
return std::auto_ptr<StoredMessage>(NULL) ;
|
||||
}
|
||||
|
||||
// ===
|
||||
@ -204,7 +66,7 @@ GSmtp::MessageStore::Iterator::Iterator() :
|
||||
{
|
||||
}
|
||||
|
||||
GSmtp::MessageStore::Iterator::Iterator( MessageStoreIteratorImp * imp ) :
|
||||
GSmtp::MessageStore::Iterator::Iterator( IteratorImp * imp ) :
|
||||
m_imp(imp)
|
||||
{
|
||||
G_ASSERT( m_imp->m_ref_count == 1UL ) ;
|
||||
@ -251,505 +113,3 @@ GSmtp::MessageStore::Iterator & GSmtp::MessageStore::Iterator::operator=( const
|
||||
return * this ;
|
||||
}
|
||||
|
||||
// ===
|
||||
|
||||
GSmtp::StoredMessage::~StoredMessage()
|
||||
{
|
||||
// empty
|
||||
}
|
||||
|
||||
// ===
|
||||
|
||||
GSmtp::NewMessage::~NewMessage()
|
||||
{
|
||||
// empty
|
||||
}
|
||||
|
||||
// ===
|
||||
|
||||
GSmtp::MessageStore * GSmtp::MessageStore::m_this = NULL ;
|
||||
|
||||
GSmtp::MessageStore::MessageStore( const G::Path & directory_path ) :
|
||||
m_imp(NULL)
|
||||
{
|
||||
if( m_this == NULL )
|
||||
m_this = this ;
|
||||
|
||||
MessageStoreImp::checkPath( directory_path ) ;
|
||||
m_imp = new MessageStoreImp( directory_path ) ;
|
||||
}
|
||||
|
||||
GSmtp::MessageStore & GSmtp::MessageStore::instance()
|
||||
{
|
||||
if( m_this == NULL )
|
||||
throw NoInstance() ;
|
||||
return * m_this ;
|
||||
}
|
||||
|
||||
GSmtp::MessageStore::~MessageStore()
|
||||
{
|
||||
if( m_this == this )
|
||||
m_this = NULL ;
|
||||
delete m_imp ;
|
||||
}
|
||||
|
||||
G::Path GSmtp::MessageStore::directory() const
|
||||
{
|
||||
return m_imp->dir() ;
|
||||
}
|
||||
|
||||
std::auto_ptr<GSmtp::NewMessage> GSmtp::MessageStore::newMessage( const std::string & from )
|
||||
{
|
||||
std::auto_ptr<NewMessage> new_message( new NewMessageImp(from,*m_imp) ) ;
|
||||
return new_message ;
|
||||
}
|
||||
|
||||
bool GSmtp::MessageStore::empty() const
|
||||
{
|
||||
return m_imp->empty() ;
|
||||
}
|
||||
|
||||
std::auto_ptr<GSmtp::StoredMessage> GSmtp::MessageStore::get()
|
||||
{
|
||||
return m_imp->get() ;
|
||||
}
|
||||
|
||||
GSmtp::MessageStore::Iterator GSmtp::MessageStore::iterator()
|
||||
{
|
||||
return m_imp->iterator() ;
|
||||
}
|
||||
|
||||
// ===
|
||||
|
||||
GSmtp::NewMessageImp::NewMessageImp( const std::string & from , MessageStoreImp & store ) :
|
||||
m_from(from) ,
|
||||
m_store(store),
|
||||
m_eight_bit(false)
|
||||
{
|
||||
m_seq = store.newSeq() ;
|
||||
|
||||
m_content_path = m_store.contentPath( m_seq ) ;
|
||||
G_LOG( "GSmtp::NewMessage: content file: " << m_content_path ) ;
|
||||
|
||||
std::auto_ptr<std::ostream> content_stream = m_store.stream( m_content_path ) ;
|
||||
m_content = content_stream ;
|
||||
}
|
||||
|
||||
GSmtp::NewMessageImp::~NewMessageImp()
|
||||
{
|
||||
}
|
||||
|
||||
void GSmtp::NewMessageImp::addTo( const std::string & to , bool local )
|
||||
{
|
||||
if( local )
|
||||
m_to_local.push_back( to ) ;
|
||||
else
|
||||
m_to_remote.push_back( to ) ;
|
||||
}
|
||||
|
||||
void GSmtp::NewMessageImp::addText( const std::string & line )
|
||||
{
|
||||
if( ! m_eight_bit )
|
||||
m_eight_bit = isEightBit(line) ;
|
||||
|
||||
*(m_content.get()) << line << crlf() ;
|
||||
}
|
||||
|
||||
bool GSmtp::NewMessageImp::isEightBit( const std::string & line )
|
||||
{
|
||||
const size_t n = line.length() ;
|
||||
for( size_t i = 0U ; i < n ; --i )
|
||||
{
|
||||
const unsigned char c = static_cast<unsigned char>(line.at(i)) ;
|
||||
if( c > 0x7fU )
|
||||
return true ;
|
||||
}
|
||||
return false ;
|
||||
}
|
||||
|
||||
void GSmtp::NewMessageImp::store()
|
||||
{
|
||||
// flush the content file
|
||||
m_content->flush() ;
|
||||
if( ! m_content->good() )
|
||||
throw GSmtp::MessageStore::WriteError( m_content_path.str() ) ;
|
||||
m_content <<= 0 ;
|
||||
|
||||
// write the envelope
|
||||
G::Path p0 = m_store.envelopeWorkingPath( m_seq ) ;
|
||||
G::Path p1 = m_store.envelopePath( m_seq ) ;
|
||||
bool ok = false ;
|
||||
{
|
||||
std::auto_ptr<std::ostream> envelope_stream = m_store.stream( p0 ) ;
|
||||
ok = saveEnvelope( *(envelope_stream.get()) , p0.str() ) ;
|
||||
}
|
||||
|
||||
// deliver to local mailboxes (in theory)
|
||||
if( ok && m_to_local.size() != 0U )
|
||||
{
|
||||
deliver( m_to_local , m_content_path , p0 , p1 ) ;
|
||||
}
|
||||
|
||||
// commit the envelope, or rollback the content
|
||||
if( ! ok || ! G::File::rename(p0,p1,G::File::NoThrow() ) )
|
||||
{
|
||||
G_ASSERT( m_content_path.str().length() != 0U ) ;
|
||||
G::File::remove( m_content_path , G::File::NoThrow() ) ;
|
||||
throw GSmtp::MessageStore::WriteError( p0.str() ) ;
|
||||
}
|
||||
}
|
||||
|
||||
void GSmtp::NewMessageImp::deliver( const G::Strings & to ,
|
||||
const G::Path & content_path , const G::Path & envelope_path_now ,
|
||||
const G::Path & envelope_path_later )
|
||||
{
|
||||
// could shell out to "procmail" or "deliver" here, but keep it
|
||||
// simple and within the scope of a "message-store" class
|
||||
|
||||
G_LOG( "GSmtp::NewMessage: copying message for local recipient(s): "
|
||||
<< content_path.basename() << ".local" ) ;
|
||||
|
||||
G::File::copy( content_path.str() , content_path.str()+".local" ) ;
|
||||
G::File::copy( envelope_path_now.str() , envelope_path_later.str()+".local" ) ;
|
||||
}
|
||||
|
||||
bool GSmtp::NewMessageImp::saveEnvelope( std::ostream & stream , const std::string & where ) const
|
||||
{
|
||||
G_LOG( "GSmtp::NewMessage: envelope file: " << where ) ;
|
||||
|
||||
const std::string x( MessageStoreImp::x() ) ;
|
||||
|
||||
stream << x << "Format: " << MessageStoreImp::format() << crlf() ;
|
||||
stream << x << "Content: " << (m_eight_bit?"8":"7") << "bit" << crlf() ;
|
||||
stream << x << "From: " << m_from << crlf() ;
|
||||
stream << x << "ToCount: " << (m_to_local.size()+m_to_remote.size()) << crlf() ;
|
||||
{
|
||||
G::Strings::const_iterator to_p = m_to_local.begin() ;
|
||||
for( ; to_p != m_to_local.end() ; ++to_p )
|
||||
stream << x << "To-Local: " << *to_p << crlf() ;
|
||||
}
|
||||
{
|
||||
G::Strings::const_iterator to_p = m_to_remote.begin() ;
|
||||
for( ; to_p != m_to_remote.end() ; ++to_p )
|
||||
stream << x << "To-Remote: " << *to_p << crlf() ;
|
||||
}
|
||||
stream << x << "End: 1" << crlf() ;
|
||||
stream.flush() ;
|
||||
return stream.good() ;
|
||||
}
|
||||
|
||||
std::string GSmtp::NewMessageImp::crlf() const
|
||||
{
|
||||
return std::string( "\015\012" ) ;
|
||||
}
|
||||
|
||||
// ===
|
||||
|
||||
GSmtp::MessageStoreImp::MessageStoreImp( const G::Path & dir ) :
|
||||
m_dir(dir),
|
||||
m_seq(1UL)
|
||||
{
|
||||
}
|
||||
|
||||
//static
|
||||
std::string GSmtp::MessageStoreImp::x()
|
||||
{
|
||||
return "X-MailRelay-" ;
|
||||
}
|
||||
|
||||
//static
|
||||
std::string GSmtp::MessageStoreImp::format()
|
||||
{
|
||||
return "#2821.2" ;
|
||||
}
|
||||
|
||||
const G::Path & GSmtp::MessageStoreImp::dir() const
|
||||
{
|
||||
return m_dir ;
|
||||
}
|
||||
|
||||
//static
|
||||
void GSmtp::MessageStoreImp::checkPath( const G::Path & directory_path )
|
||||
{
|
||||
// (void) G::File::mkdir( directory_path ) ;
|
||||
G::Directory dir_test( directory_path ) ;
|
||||
if( ! dir_test.valid() )
|
||||
{
|
||||
throw MessageStore::InvalidDirectory( directory_path.str() ) ;
|
||||
}
|
||||
if( ! dir_test.valid(true) )
|
||||
{
|
||||
G_WARNING( "GSmtp::MessageStore: "
|
||||
<< "directory not writable: \""
|
||||
<< directory_path << "\"" ) ;
|
||||
}
|
||||
}
|
||||
|
||||
std::auto_ptr<std::ostream> GSmtp::MessageStoreImp::stream( const G::Path & path )
|
||||
{
|
||||
std::auto_ptr<std::ostream> ptr(
|
||||
new std::ofstream( path.pathCstr() ,
|
||||
std::ios_base::binary | std::ios_base::out | std::ios_base::trunc ) ) ;
|
||||
return ptr ;
|
||||
}
|
||||
|
||||
G::Path GSmtp::MessageStoreImp::contentPath( unsigned long seq ) const
|
||||
{
|
||||
return fullPath( filePrefix(seq) + ".content" ) ;
|
||||
}
|
||||
|
||||
G::Path GSmtp::MessageStoreImp::envelopePath( unsigned long seq ) const
|
||||
{
|
||||
return fullPath( filePrefix(seq) + ".envelope" ) ;
|
||||
}
|
||||
|
||||
G::Path GSmtp::MessageStoreImp::envelopeWorkingPath( unsigned long seq ) const
|
||||
{
|
||||
return fullPath( filePrefix(seq) + ".envelope.new" ) ;
|
||||
}
|
||||
|
||||
std::string GSmtp::MessageStoreImp::filePrefix( unsigned long seq ) const
|
||||
{
|
||||
std::stringstream ss ;
|
||||
ss << "emailrelay." << G::Pid() << "." << seq ;
|
||||
return ss.str() ;
|
||||
}
|
||||
|
||||
G::Path GSmtp::MessageStoreImp::fullPath( const std::string & filename ) const
|
||||
{
|
||||
G::Path p( m_dir ) ;
|
||||
p.pathAppend( filename ) ;
|
||||
return p ;
|
||||
}
|
||||
|
||||
unsigned long GSmtp::MessageStoreImp::newSeq()
|
||||
{
|
||||
return m_seq++ ;
|
||||
}
|
||||
|
||||
bool GSmtp::MessageStoreImp::empty() const
|
||||
{
|
||||
G::Directory dir(m_dir) ;
|
||||
G::DirectoryIterator iter( dir , "*.envelope" ) ;
|
||||
bool no_more = iter.error() || !iter.more() ;
|
||||
if( no_more )
|
||||
G_DEBUG( "GSmtp::MessageStoreImp: no files to process" ) ;
|
||||
return no_more ;
|
||||
}
|
||||
|
||||
GSmtp::MessageStore::Iterator GSmtp::MessageStoreImp::iterator()
|
||||
{
|
||||
return MessageStore::Iterator( new MessageStoreIteratorImp(G::Directory(m_dir)) ) ;
|
||||
}
|
||||
|
||||
std::auto_ptr<GSmtp::StoredMessage> GSmtp::MessageStoreImp::get()
|
||||
{
|
||||
MessageStore::Iterator iter = iterator() ;
|
||||
return iter.next() ;
|
||||
}
|
||||
|
||||
// ===
|
||||
|
||||
GSmtp::StoredMessageImp::StoredMessageImp( const G::Path & path ) :
|
||||
m_envelope_path(path)
|
||||
{
|
||||
G_DEBUG( "StoredMessageImp: \"" << path << "\"" ) ;
|
||||
}
|
||||
|
||||
GSmtp::StoredMessageImp::~StoredMessageImp()
|
||||
{
|
||||
}
|
||||
|
||||
bool GSmtp::StoredMessageImp::eightBit() const
|
||||
{
|
||||
return m_eight_bit ;
|
||||
}
|
||||
|
||||
bool GSmtp::StoredMessageImp::readEnvelope( std::string & reason )
|
||||
{
|
||||
try
|
||||
{
|
||||
return readEnvelopeCore( reason ) ;
|
||||
}
|
||||
catch( std::exception & e )
|
||||
{
|
||||
reason = e.what() ;
|
||||
return false ;
|
||||
}
|
||||
}
|
||||
|
||||
bool GSmtp::StoredMessageImp::readEnvelopeCore( std::string & reason )
|
||||
{
|
||||
std::ifstream stream( m_envelope_path.str().c_str() , std::ios_base::binary | std::ios_base::in ) ;
|
||||
if( ! stream.good() )
|
||||
throw OpenError() ;
|
||||
|
||||
std::string format_line = getline(stream) ;
|
||||
if( value(format_line) != MessageStoreImp::format() )
|
||||
throw InvalidFormat( value(format_line) + "!=" + MessageStoreImp::format() ) ;
|
||||
|
||||
std::string content_line = getline(stream) ;
|
||||
m_eight_bit = value(content_line) == "8bit" ;
|
||||
|
||||
m_from = value(getline(stream)) ;
|
||||
G_DEBUG( "GSmtp::StoredMessageImp::readEnvelope: from \"" << m_from << "\"" ) ;
|
||||
|
||||
std::string to_count_line = getline(stream) ;
|
||||
unsigned int to_count = G::Str::toUInt( value(to_count_line) ) ;
|
||||
|
||||
for( unsigned int i = 0U ; i < to_count ; i++ )
|
||||
{
|
||||
std::string to_line = getline(stream) ;
|
||||
bool is_local = to_line.find(MessageStoreImp::x()+"To-Local") == 0U ;
|
||||
bool is_remote = to_line.find(MessageStoreImp::x()+"To-Remote") == 0U ;
|
||||
if( ! is_local && ! is_remote )
|
||||
throw InvalidTo(to_line) ;
|
||||
|
||||
G_DEBUG( "GSmtp::StoredMessageImp::readEnvelope: to "
|
||||
"[" << (i+1U) << "/" << to_count << "] "
|
||||
"(" << (is_local?"local":"remote") << ") "
|
||||
<< "\"" << value(to_line) << "\"" ) ;
|
||||
|
||||
if( is_local )
|
||||
m_to_local.push_back( value(to_line) ) ;
|
||||
else
|
||||
m_to_remote.push_back( value(to_line) ) ;
|
||||
}
|
||||
|
||||
std::string end = getline(stream) ;
|
||||
if( end.find(MessageStoreImp::x()+"End") != 0U )
|
||||
throw NoEnd() ;
|
||||
|
||||
if( m_to_remote.size() == 0U )
|
||||
throw NoRecipients() ;
|
||||
|
||||
if( ! stream.good() )
|
||||
throw StreamError() ;
|
||||
|
||||
return stream.good() ;
|
||||
}
|
||||
|
||||
bool GSmtp::StoredMessageImp::openContent( std::string & reason )
|
||||
{
|
||||
try
|
||||
{
|
||||
G::Path content_path = contentPath() ;
|
||||
G_DEBUG( "GSmtp::MessageStoreImp::openContent: \"" << content_path << "\"" ) ;
|
||||
std::auto_ptr<std::istream> stream( new std::ifstream(
|
||||
content_path.str().c_str() , std::ios_base::in | std::ios_base::binary ) ) ;
|
||||
if( !stream->good() )
|
||||
{
|
||||
reason = "cannot open content file" ;
|
||||
return false ;
|
||||
}
|
||||
|
||||
G_LOG( "GSmtp::MessageStore: processing envelope \"" << m_envelope_path.basename() << "\"" ) ;
|
||||
G_LOG( "GSmtp::MessageStore: processing content \"" << content_path.basename() << "\"" ) ;
|
||||
|
||||
m_content = stream ;
|
||||
return true ;
|
||||
}
|
||||
catch( std::exception & e )
|
||||
{
|
||||
G_WARNING( "GSmtp::MessageStoreImp: exception: " << e.what() ) ;
|
||||
reason = e.what() ;
|
||||
return false ;
|
||||
}
|
||||
}
|
||||
|
||||
std::string GSmtp::StoredMessageImp::getline( std::istream & stream ) const
|
||||
{
|
||||
return G::Str::readLineFrom( stream , crlf() ) ;
|
||||
}
|
||||
|
||||
std::string GSmtp::StoredMessageImp::value( const std::string & s ) const
|
||||
{
|
||||
size_t pos = s.find(' ') ;
|
||||
if( pos == std::string::npos )
|
||||
throw MessageStore::FormatError() ;
|
||||
return s.substr(pos+1U) ;
|
||||
}
|
||||
|
||||
std::string GSmtp::StoredMessageImp::crlf() const
|
||||
{
|
||||
return std::string( "\015\012" ) ;
|
||||
}
|
||||
|
||||
bool GSmtp::StoredMessageImp::lock()
|
||||
{
|
||||
G::Path & src = m_envelope_path ;
|
||||
G::Path dst( src.str() + ".busy" ) ;
|
||||
bool ok = G::File::rename( src , dst , G::File::NoThrow() ) ;
|
||||
if( ok )
|
||||
{
|
||||
G_LOG( "GSmtp::StoredMessage: locking file \"" << src.basename() << "\"" ) ;
|
||||
m_envelope_path = dst ;
|
||||
}
|
||||
return ok ;
|
||||
}
|
||||
|
||||
void GSmtp::StoredMessageImp::fail( const std::string & reason )
|
||||
{
|
||||
try
|
||||
{
|
||||
{
|
||||
std::ofstream file( m_envelope_path.str().c_str() ,
|
||||
std::ios_base::binary | std::ios_base::ate ) ;
|
||||
file << MessageStoreImp::x() << "Reason: " << reason ;
|
||||
}
|
||||
|
||||
G::Path env_temp( m_envelope_path ) ; // "foo.envelope.busy"
|
||||
env_temp.removeExtension() ; // "foo.envelope"
|
||||
G::Path bad( env_temp.str() + ".bad" ) ; // "foo.envelope.bad"
|
||||
G_LOG( "GSmtp::StoredMessage: failing file: "
|
||||
<< "\"" << m_envelope_path.basename() << "\" -> "
|
||||
<< "\"" << bad.basename() << "\"" ) ;
|
||||
|
||||
G::File::rename( m_envelope_path , bad , G::File::NoThrow() ) ;
|
||||
}
|
||||
catch(...)
|
||||
{
|
||||
}
|
||||
}
|
||||
|
||||
void GSmtp::StoredMessageImp::destroy()
|
||||
{
|
||||
try
|
||||
{
|
||||
G_LOG( "GSmtp::StoredMessage: deleting file: \"" << m_envelope_path.basename() << "\"" ) ;
|
||||
G::File::remove( m_envelope_path , G::File::NoThrow() ) ;
|
||||
|
||||
G::Path content_path = contentPath() ;
|
||||
G_LOG( "GSmtp::StoredMessage: deleting file: \"" << content_path.basename() << "\"" ) ;
|
||||
m_content <<= 0 ; // close it first
|
||||
G::File::remove( content_path , G::File::NoThrow() ) ;
|
||||
}
|
||||
catch(...)
|
||||
{
|
||||
}
|
||||
}
|
||||
|
||||
const std::string & GSmtp::StoredMessageImp::from() const
|
||||
{
|
||||
return m_from ;
|
||||
}
|
||||
|
||||
const G::Strings & GSmtp::StoredMessageImp::to() const
|
||||
{
|
||||
return m_to_remote ;
|
||||
}
|
||||
|
||||
std::auto_ptr<std::istream> GSmtp::StoredMessageImp::extractContentStream()
|
||||
{
|
||||
G_ASSERT( m_content.get() != NULL ) ;
|
||||
return m_content ;
|
||||
}
|
||||
|
||||
G::Path GSmtp::StoredMessageImp::contentPath() const
|
||||
{
|
||||
G::Path path( m_envelope_path ) ; // "foo.envelope.busy"
|
||||
path.removeExtension() ; // "foo.envelope"
|
||||
path.setExtension( "content" ) ; // "foo.content"
|
||||
return path ;
|
||||
}
|
||||
|
||||
|
@ -26,6 +26,8 @@
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gnewmessage.h"
|
||||
#include "gstoredmessage.h"
|
||||
#include "gexception.h"
|
||||
#include "gstrings.h"
|
||||
#include "gpath.h"
|
||||
@ -33,73 +35,6 @@
|
||||
namespace GSmtp
|
||||
{
|
||||
class MessageStore ;
|
||||
class StoredMessage ;
|
||||
class NewMessage ;
|
||||
|
||||
class MessageStoreImp ;
|
||||
class MessageStoreIteratorImp ;
|
||||
class StoredMessageImp ;
|
||||
class NewMessageImp ;
|
||||
} ;
|
||||
|
||||
// Class: GSmtp::NewMessage
|
||||
// Description: An abstract class to allow the creation
|
||||
// of a new message in the message store.
|
||||
// See also: MessageStore, MessageStore::newMessage()
|
||||
//
|
||||
class GSmtp::NewMessage
|
||||
{
|
||||
public:
|
||||
virtual void addTo( const std::string & to , bool local ) = 0 ;
|
||||
// Adds a 'to' address.
|
||||
|
||||
virtual void addText( const std::string & line ) = 0 ;
|
||||
// Adds a line of content.
|
||||
|
||||
virtual void store() = 0 ;
|
||||
// Stores the message in the message store.
|
||||
|
||||
virtual ~NewMessage() ;
|
||||
// Destructor.
|
||||
|
||||
private:
|
||||
void operator=( const NewMessage & ) ;
|
||||
} ;
|
||||
|
||||
// Class: GSmtp::StoredMessage
|
||||
// Description: An abstract class for messages which have
|
||||
// come from the store.
|
||||
// See also: MessageStore, MessageStore::get()
|
||||
//
|
||||
class GSmtp::StoredMessage
|
||||
{
|
||||
public:
|
||||
virtual const std::string & from() const = 0 ;
|
||||
// Returns the envelope 'from' field.
|
||||
|
||||
virtual const G::Strings & to() const = 0 ;
|
||||
// Returns the envelope 'to' fields.
|
||||
|
||||
virtual std::auto_ptr<std::istream> extractContentStream() = 0 ;
|
||||
// Extracts the content stream.
|
||||
// Can only be called once.
|
||||
|
||||
virtual void destroy() = 0 ;
|
||||
// Deletes the message within the store.
|
||||
|
||||
virtual void fail( const std::string & reason ) = 0 ;
|
||||
// Marks the message as failed within the store.
|
||||
|
||||
virtual bool eightBit() const = 0 ;
|
||||
// Returns true if the message content (header+body)
|
||||
// contains a character with the most significant
|
||||
// bit set.
|
||||
|
||||
virtual ~StoredMessage() ;
|
||||
// Destructor.
|
||||
|
||||
private:
|
||||
void operator=( const StoredMessage & ) ;
|
||||
} ;
|
||||
|
||||
// Class: GSmtp::MessageStore
|
||||
@ -118,16 +53,24 @@ private:
|
||||
class GSmtp::MessageStore
|
||||
{
|
||||
public:
|
||||
G_EXCEPTION( InvalidDirectory , "invalid spool directory" ) ;
|
||||
G_EXCEPTION( WriteError , "error writing file" ) ;
|
||||
G_EXCEPTION( NoInstance , "no message store instance" ) ;
|
||||
G_EXCEPTION( FormatError , "format error" ) ;
|
||||
class IteratorImp // A base class for MessageStore::Iterator implementations.
|
||||
{
|
||||
public: unsigned long m_ref_count ;
|
||||
public: virtual std::auto_ptr<GSmtp::StoredMessage> next() = 0 ;
|
||||
public: IteratorImp() ;
|
||||
public: virtual ~IteratorImp() ;
|
||||
private: IteratorImp( const IteratorImp & ) ;
|
||||
private: void operator=( const IteratorImp & ) ;
|
||||
} ;
|
||||
class Iterator // An iterator class for GSmtp::MessageStore.
|
||||
{
|
||||
public: std::auto_ptr<StoredMessage> next() ;
|
||||
private: MessageStoreIteratorImp * m_imp ;
|
||||
private: IteratorImp * m_imp ;
|
||||
public: Iterator() ;
|
||||
public: explicit Iterator( MessageStoreIteratorImp * ) ;
|
||||
public: explicit Iterator( IteratorImp * ) ;
|
||||
public: ~Iterator() ;
|
||||
public: Iterator( const Iterator & ) ;
|
||||
public: Iterator & operator=( const Iterator & ) ;
|
||||
@ -141,29 +84,29 @@ public:
|
||||
// "/usr/local/var/spool/emailrelay". (Typically
|
||||
// has an os-specific implementation.)
|
||||
|
||||
explicit MessageStore( const G::Path & directory ) ;
|
||||
// Constructor. Throws exceptions if
|
||||
// not a valid storage directory.
|
||||
MessageStore() ;
|
||||
// Default constructor.
|
||||
|
||||
~MessageStore() ;
|
||||
virtual ~MessageStore() ;
|
||||
// Destructor.
|
||||
|
||||
std::auto_ptr<NewMessage> newMessage( const std::string & from ) ;
|
||||
virtual std::auto_ptr<NewMessage> newMessage( const std::string & from ) = 0 ;
|
||||
// Creates a new message.
|
||||
|
||||
bool empty() const ;
|
||||
virtual bool empty() const = 0 ;
|
||||
// Returns true if the message store is empty.
|
||||
|
||||
std::auto_ptr<StoredMessage> get() ;
|
||||
// Pulls a message out of the store (selected
|
||||
// at random). Returns a NULL smart pointer
|
||||
// if there are no messages to extract.
|
||||
virtual std::auto_ptr<StoredMessage> get( unsigned long id ) = 0 ;
|
||||
// Pulls a message out of the store.
|
||||
// Throws execptions on error.
|
||||
//
|
||||
// See also NewMessage::id().
|
||||
//
|
||||
// As a side effect some stored messages may be
|
||||
// marked as bad, or deleted (if they
|
||||
// have no recipients).
|
||||
|
||||
Iterator iterator() ;
|
||||
virtual Iterator iterator() = 0 ;
|
||||
// Returns a read iterator. (Note that copies of
|
||||
// iterators share state. For independent iterators
|
||||
// call iterator() for each.)
|
||||
@ -172,17 +115,12 @@ public:
|
||||
// messages may be marked as bad, or deleted (if
|
||||
// they have no recipients).
|
||||
|
||||
G::Path directory() const ;
|
||||
// Returns the storage directory (as passed
|
||||
// to the constructor).
|
||||
|
||||
private:
|
||||
MessageStore( const MessageStore & ) ;
|
||||
void operator=( const MessageStore & ) ;
|
||||
|
||||
private:
|
||||
static MessageStore * m_this ;
|
||||
MessageStoreImp * m_imp ;
|
||||
} ;
|
||||
|
||||
#endif
|
||||
|
207
src/main/gnewfile.cpp
Normal file
207
src/main/gnewfile.cpp
Normal file
@ -0,0 +1,207 @@
|
||||
//
|
||||
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
//
|
||||
// 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, write to the Free Software
|
||||
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gnewfile.cpp
|
||||
//
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gmessagestore.h"
|
||||
#include "gnewfile.h"
|
||||
#include "gmemory.h"
|
||||
#include "gprocess.h"
|
||||
#include "gfile.h"
|
||||
#include "gassert.h"
|
||||
#include "glog.h"
|
||||
#include <iostream>
|
||||
#include <fstream>
|
||||
|
||||
bool GSmtp::NewFile::m_preprocess = false ;
|
||||
G::Path GSmtp::NewFile::m_preprocessor ;
|
||||
|
||||
GSmtp::NewFile::NewFile( const std::string & from , FileStore & store ) :
|
||||
m_from(from) ,
|
||||
m_store(store),
|
||||
m_eight_bit(false)
|
||||
{
|
||||
m_seq = store.newSeq() ;
|
||||
|
||||
m_content_path = m_store.contentPath( m_seq ) ;
|
||||
G_LOG( "GSmtp::NewMessage: content file: " << m_content_path ) ;
|
||||
|
||||
std::auto_ptr<std::ostream> content_stream = m_store.stream( m_content_path ) ;
|
||||
m_content = content_stream ;
|
||||
}
|
||||
|
||||
GSmtp::NewFile::~NewFile()
|
||||
{
|
||||
}
|
||||
|
||||
void GSmtp::NewFile::addTo( const std::string & to , bool local )
|
||||
{
|
||||
if( local )
|
||||
m_to_local.push_back( to ) ;
|
||||
else
|
||||
m_to_remote.push_back( to ) ;
|
||||
}
|
||||
|
||||
void GSmtp::NewFile::addText( const std::string & line )
|
||||
{
|
||||
if( ! m_eight_bit )
|
||||
m_eight_bit = isEightBit(line) ;
|
||||
|
||||
*(m_content.get()) << line << crlf() ;
|
||||
}
|
||||
|
||||
bool GSmtp::NewFile::isEightBit( const std::string & line )
|
||||
{
|
||||
const size_t n = line.length() ;
|
||||
for( size_t i = 0U ; i < n ; --i )
|
||||
{
|
||||
const unsigned char c = static_cast<unsigned char>(line.at(i)) ;
|
||||
if( c > 0x7fU )
|
||||
return true ;
|
||||
}
|
||||
return false ;
|
||||
}
|
||||
|
||||
void GSmtp::NewFile::store()
|
||||
{
|
||||
// flush the content file
|
||||
//
|
||||
m_content->flush() ;
|
||||
if( ! m_content->good() )
|
||||
throw GSmtp::MessageStore::WriteError( m_content_path.str() ) ;
|
||||
m_content <<= 0 ;
|
||||
|
||||
// write the envelope
|
||||
//
|
||||
G::Path p0 = m_store.envelopeWorkingPath( m_seq ) ;
|
||||
G::Path p1 = m_store.envelopePath( m_seq ) ;
|
||||
bool ok = false ;
|
||||
std::string reason = p0.str() ;
|
||||
{
|
||||
std::auto_ptr<std::ostream> envelope_stream = m_store.stream( p0 ) ;
|
||||
ok = saveEnvelope( *(envelope_stream.get()) , p0.str() ) ;
|
||||
}
|
||||
|
||||
// shell out to a message pre-processor
|
||||
//
|
||||
if( ok )
|
||||
{
|
||||
ok = preprocess( m_content_path ) ;
|
||||
if( !ok )
|
||||
reason = "pre-processing failed" ;
|
||||
}
|
||||
|
||||
// deliver to local mailboxes
|
||||
//
|
||||
if( ok && m_to_local.size() != 0U )
|
||||
{
|
||||
deliver( m_to_local , m_content_path , p0 , p1 ) ;
|
||||
}
|
||||
|
||||
// commit the envelope, or rollback the content
|
||||
//
|
||||
if( ! ok || ! G::File::rename(p0,p1,G::File::NoThrow() ) )
|
||||
{
|
||||
G_ASSERT( m_content_path.str().length() != 0U ) ;
|
||||
G::File::remove( m_content_path , G::File::NoThrow() ) ;
|
||||
throw GSmtp::MessageStore::WriteError( reason ) ;
|
||||
}
|
||||
}
|
||||
|
||||
bool GSmtp::NewFile::preprocess( const G::Path & path )
|
||||
{
|
||||
if( m_preprocess )
|
||||
{
|
||||
G_LOG( "GSmtp::NewFile::preprocess: " << m_preprocessor << " " << path ) ;
|
||||
int exit_code = G::Process::spawn( m_preprocessor , path.str() ) ;
|
||||
G_LOG( "GSmtp::NewFile::preprocess: exit status " << exit_code ) ;
|
||||
if( exit_code != 0 )
|
||||
{
|
||||
G_WARNING( "GSmtp::NewFile::preprocess: pre-processing failed: exit code " << exit_code ) ;
|
||||
return false ;
|
||||
}
|
||||
}
|
||||
return true ;
|
||||
}
|
||||
|
||||
void GSmtp::NewFile::deliver( const G::Strings & to ,
|
||||
const G::Path & content_path , const G::Path & envelope_path_now ,
|
||||
const G::Path & envelope_path_later )
|
||||
{
|
||||
// could shell out to "procmail" or "deliver" here, but keep it
|
||||
// simple and within the scope of a "message-store" class
|
||||
|
||||
G_LOG( "GSmtp::NewMessage: copying message for local recipient(s): "
|
||||
<< content_path.basename() << ".local" ) ;
|
||||
|
||||
G::File::copy( content_path.str() , content_path.str()+".local" ) ;
|
||||
G::File::copy( envelope_path_now.str() , envelope_path_later.str()+".local" ) ;
|
||||
}
|
||||
|
||||
bool GSmtp::NewFile::saveEnvelope( std::ostream & stream , const std::string & where ) const
|
||||
{
|
||||
G_LOG( "GSmtp::NewMessage: envelope file: " << where ) ;
|
||||
|
||||
const std::string x( m_store.x() ) ;
|
||||
|
||||
stream << x << "Format: " << m_store.format() << crlf() ;
|
||||
stream << x << "Content: " << (m_eight_bit?"8":"7") << "bit" << crlf() ;
|
||||
stream << x << "From: " << m_from << crlf() ;
|
||||
stream << x << "ToCount: " << (m_to_local.size()+m_to_remote.size()) << crlf() ;
|
||||
{
|
||||
G::Strings::const_iterator to_p = m_to_local.begin() ;
|
||||
for( ; to_p != m_to_local.end() ; ++to_p )
|
||||
stream << x << "To-Local: " << *to_p << crlf() ;
|
||||
}
|
||||
{
|
||||
G::Strings::const_iterator to_p = m_to_remote.begin() ;
|
||||
for( ; to_p != m_to_remote.end() ; ++to_p )
|
||||
stream << x << "To-Remote: " << *to_p << crlf() ;
|
||||
}
|
||||
stream << x << "End: 1" << crlf() ;
|
||||
stream.flush() ;
|
||||
return stream.good() ;
|
||||
}
|
||||
|
||||
std::string GSmtp::NewFile::crlf() const
|
||||
{
|
||||
return std::string( "\015\012" ) ;
|
||||
}
|
||||
|
||||
unsigned long GSmtp::NewFile::id() const
|
||||
{
|
||||
return m_seq ;
|
||||
}
|
||||
|
||||
void GSmtp::NewFile::setPreprocessor( const G::Path & exe )
|
||||
{
|
||||
if( exe.isRelative() )
|
||||
throw InvalidPath( exe.str() ) ;
|
||||
|
||||
if( G::Process::privileged() )
|
||||
throw Dangerous() ;
|
||||
|
||||
m_preprocess = true ;
|
||||
m_preprocessor = exe ;
|
||||
}
|
||||
|
81
src/main/gnewfile.h
Normal file
81
src/main/gnewfile.h
Normal file
@ -0,0 +1,81 @@
|
||||
//
|
||||
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
//
|
||||
// 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, write to the Free Software
|
||||
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gnewfile.h
|
||||
//
|
||||
|
||||
#ifndef G_SMTP_NEW_FILE_H
|
||||
#define G_SMTP_NEW_FILE_H
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gfilestore.h"
|
||||
#include "gnewmessage.h"
|
||||
#include "gexception.h"
|
||||
#include <iostream>
|
||||
|
||||
namespace GSmtp
|
||||
{
|
||||
class NewFile ;
|
||||
} ;
|
||||
|
||||
// Class: GSmtp::NewFile
|
||||
// Description: A concrete derived class implementing the
|
||||
// NewMessage interface. Writes itself to the i/o streams
|
||||
// supplied by MessageStoreImp.
|
||||
//
|
||||
class GSmtp::NewFile : public GSmtp:: NewMessage
|
||||
{
|
||||
public:
|
||||
G_EXCEPTION( InvalidPath , "invalid path -- must be absolute" ) ;
|
||||
G_EXCEPTION( Dangerous , "message filtering not allowed if running as a privileged process" ) ;
|
||||
|
||||
NewFile( const std::string & from , FileStore & store ) ;
|
||||
virtual ~NewFile() ;
|
||||
virtual void addTo( const std::string & to , bool local ) ;
|
||||
virtual void addText( const std::string & line ) ;
|
||||
virtual void store() ;
|
||||
virtual unsigned long id() const ;
|
||||
|
||||
static void setPreprocessor( const G::Path & exe ) ;
|
||||
// Defines a program which is used for pre-processing
|
||||
// messages before they are stored.
|
||||
|
||||
private:
|
||||
FileStore & m_store ;
|
||||
unsigned long m_seq ;
|
||||
std::string m_from ;
|
||||
G::Strings m_to_local ;
|
||||
G::Strings m_to_remote ;
|
||||
std::auto_ptr<std::ostream> m_content ;
|
||||
G::Path m_content_path ;
|
||||
bool m_eight_bit ;
|
||||
static bool m_preprocess ;
|
||||
static G::Path m_preprocessor ;
|
||||
|
||||
private:
|
||||
bool saveEnvelope( std::ostream & stream , const std::string & where ) const ;
|
||||
std::string crlf() const ;
|
||||
static bool isEightBit( const std::string & line ) ;
|
||||
void deliver( const G::Strings & , const G::Path & , const G::Path & , const G::Path & ) ;
|
||||
bool preprocess( const G::Path & ) ;
|
||||
} ;
|
||||
|
||||
#endif
|
@ -18,62 +18,17 @@
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gpid_win32.cpp
|
||||
// gnewmessage.cpp
|
||||
//
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gpid.h"
|
||||
#include <process.h>
|
||||
#include "gsmtp.h"
|
||||
#include "gmessagestore.h"
|
||||
#include "gnewmessage.h"
|
||||
#include <iostream>
|
||||
|
||||
namespace G
|
||||
GSmtp::NewMessage::~NewMessage()
|
||||
{
|
||||
class PidImp ;
|
||||
} ;
|
||||
|
||||
// Class: G::PidImp
|
||||
// Description: A pimple implementation class for GPid.
|
||||
//
|
||||
class G::PidImp
|
||||
{
|
||||
public:
|
||||
int m_pid ;
|
||||
} ;
|
||||
|
||||
// ===
|
||||
|
||||
G::Pid::Pid() : m_imp(NULL)
|
||||
{
|
||||
m_imp = new PidImp ;
|
||||
m_imp->m_pid = ::_getpid() ;
|
||||
// empty
|
||||
}
|
||||
|
||||
G::Pid::~Pid()
|
||||
{
|
||||
delete m_imp ;
|
||||
}
|
||||
|
||||
G::Pid::Pid( const Pid & other ) :
|
||||
m_imp(NULL)
|
||||
{
|
||||
m_imp = new PidImp ;
|
||||
m_imp->m_pid = other.m_imp->m_pid ;
|
||||
}
|
||||
|
||||
G::Pid & G::Pid::operator=( const Pid & rhs )
|
||||
{
|
||||
m_imp->m_pid = rhs.m_imp->m_pid ;
|
||||
return *this ;
|
||||
}
|
||||
|
||||
std::string G::Pid::str() const
|
||||
{
|
||||
std::stringstream ss ;
|
||||
ss << m_imp->m_pid ;
|
||||
return ss.str() ;
|
||||
}
|
||||
|
||||
bool G::Pid::operator==( const Pid & rhs ) const
|
||||
{
|
||||
return m_imp->m_pid == rhs.m_imp->m_pid ;
|
||||
} ;
|
||||
|
@ -18,59 +18,46 @@
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gpid.h
|
||||
// gnewmessage.h
|
||||
//
|
||||
|
||||
#ifndef G_PID_H
|
||||
#define G_PID_H
|
||||
#ifndef G_SMTP_NEW_MESSAGE_H
|
||||
#define G_SMTP_NEW_MESSAGE_H
|
||||
|
||||
#include "gdef.h"
|
||||
#include <iostream>
|
||||
#include "gsmtp.h"
|
||||
|
||||
namespace G
|
||||
namespace GSmtp
|
||||
{
|
||||
class PidImp ;
|
||||
class Pid ;
|
||||
class NewMessage ;
|
||||
class MessageStoreImp ;
|
||||
} ;
|
||||
|
||||
// Class: G::Pid
|
||||
// Description: A process-id class. Uses a pimple
|
||||
// pattern to hide windows/unix type differences.
|
||||
// Class: GSmtp::NewMessage
|
||||
// Description: An abstract class to allow the creation
|
||||
// of a new message in the message store.
|
||||
// See also: MessageStore, MessageStore::newMessage()
|
||||
//
|
||||
class G::Pid
|
||||
class GSmtp::NewMessage
|
||||
{
|
||||
public:
|
||||
Pid() ;
|
||||
// Default constructor for this
|
||||
// process's id.
|
||||
virtual void addTo( const std::string & to , bool local ) = 0 ;
|
||||
// Adds a 'to' address.
|
||||
|
||||
~Pid() ;
|
||||
virtual void addText( const std::string & line ) = 0 ;
|
||||
// Adds a line of content.
|
||||
|
||||
virtual void store() = 0 ;
|
||||
// Stores the message in the message store.
|
||||
|
||||
virtual unsigned long id() const = 0 ;
|
||||
// Returns the message's unique identifier.
|
||||
|
||||
virtual ~NewMessage() ;
|
||||
// Destructor.
|
||||
|
||||
Pid( const Pid & ) ;
|
||||
// Copy constructor.
|
||||
|
||||
Pid & operator=( const Pid & ) ;
|
||||
// Assignment operator.
|
||||
|
||||
std::string str() const ;
|
||||
// Returns a string representation.
|
||||
|
||||
bool operator==( const Pid & ) const ;
|
||||
// Comparison operator.
|
||||
|
||||
private:
|
||||
PidImp * m_imp ;
|
||||
} ;
|
||||
|
||||
namespace G
|
||||
{
|
||||
inline
|
||||
std::ostream & operator<<( std::ostream & stream , const Pid & pid )
|
||||
{
|
||||
stream << pid.str() ;
|
||||
return stream ;
|
||||
}
|
||||
void operator=( const NewMessage & ) ; // not implemented
|
||||
} ;
|
||||
|
||||
#endif
|
@ -24,147 +24,14 @@
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gprotocolmessage.h"
|
||||
#include "gmessagestore.h"
|
||||
#include "gmemory.h"
|
||||
#include "gstr.h"
|
||||
#include "gassert.h"
|
||||
#include "glog.h"
|
||||
|
||||
// Class: GSmtp::ProtocolMessageImp
|
||||
// Description: A private pimple-pattern implementation class for GSmtp::ProtocolMessage.
|
||||
//
|
||||
class GSmtp::ProtocolMessageImp
|
||||
{
|
||||
public:
|
||||
std::auto_ptr<NewMessage> m_msg ;
|
||||
} ;
|
||||
|
||||
// ===
|
||||
|
||||
GSmtp::ProtocolMessage::ProtocolMessage() :
|
||||
m_imp(NULL)
|
||||
{
|
||||
m_imp = new ProtocolMessageImp ;
|
||||
}
|
||||
|
||||
GSmtp::ProtocolMessage::~ProtocolMessage()
|
||||
{
|
||||
delete m_imp ;
|
||||
}
|
||||
|
||||
void GSmtp::ProtocolMessage::clear()
|
||||
// ===
|
||||
|
||||
GSmtp::ProtocolMessage::Callback::~Callback()
|
||||
{
|
||||
m_imp->m_msg <<= 0 ;
|
||||
}
|
||||
|
||||
bool GSmtp::ProtocolMessage::setFrom( const std::string & from )
|
||||
{
|
||||
try
|
||||
{
|
||||
if( from.length() == 0U )
|
||||
return false ;
|
||||
|
||||
G_ASSERT( m_imp->m_msg.get() == NULL ) ;
|
||||
clear() ; // just in case
|
||||
std::auto_ptr<NewMessage> new_msg = MessageStore::instance().newMessage( from ) ;
|
||||
m_imp->m_msg = new_msg ;
|
||||
return true ;
|
||||
}
|
||||
catch( std::exception & e )
|
||||
{
|
||||
G_ERROR( "GSmtp::ProtocolMessage::setFrom: error: " << e.what() ) ;
|
||||
return false ;
|
||||
}
|
||||
}
|
||||
|
||||
bool GSmtp::ProtocolMessage::addTo( const std::string & to )
|
||||
{
|
||||
G_ASSERT( m_imp->m_msg.get() != NULL ) ;
|
||||
if( to.length() > 0U && m_imp->m_msg.get() != NULL )
|
||||
{
|
||||
bool is_local = isLocal(to) ;
|
||||
if( is_local && !isValid(to) )
|
||||
{
|
||||
G_WARNING( "GSmtp::ProtocolMessage: rejecting local recipent (not postmaster): " << to ) ;
|
||||
return false ;
|
||||
}
|
||||
else
|
||||
{
|
||||
m_imp->m_msg->addTo( to , is_local ) ;
|
||||
return true ;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
return false ;
|
||||
}
|
||||
}
|
||||
|
||||
void GSmtp::ProtocolMessage::addReceived( const std::string & line )
|
||||
{
|
||||
addText( line ) ;
|
||||
}
|
||||
|
||||
void GSmtp::ProtocolMessage::addText( const std::string & line )
|
||||
{
|
||||
G_ASSERT( m_imp->m_msg.get() != NULL ) ;
|
||||
if( m_imp->m_msg.get() != NULL )
|
||||
m_imp->m_msg->addText( line ) ;
|
||||
}
|
||||
|
||||
std::string GSmtp::ProtocolMessage::process()
|
||||
{
|
||||
try
|
||||
{
|
||||
G_ASSERT( m_imp->m_msg.get() != NULL ) ;
|
||||
if( m_imp->m_msg.get() != NULL )
|
||||
{
|
||||
m_imp->m_msg->store() ;
|
||||
}
|
||||
clear() ;
|
||||
return std::string() ;
|
||||
}
|
||||
catch( std::exception & e )
|
||||
{
|
||||
G_ERROR( "GSmtp::ProtocolMessage::process: error: " << e.what() ) ;
|
||||
clear() ;
|
||||
return std::string( e.what() ) ;
|
||||
}
|
||||
}
|
||||
|
||||
std::pair<bool,std::string> GSmtp::ProtocolMessage::verify( const std::string & user )
|
||||
{
|
||||
G_DEBUG( "GSmtp::ProtocolMessage::verify: \"" << user << "\"" ) ;
|
||||
std::pair<bool,std::string> rc( isLocal(user) , std::string() ) ;
|
||||
if( isLocal(user) && isValid(user) )
|
||||
rc.second = fullName(user) ;
|
||||
return rc ;
|
||||
}
|
||||
|
||||
//static
|
||||
bool GSmtp::ProtocolMessage::isLocal( const std::string & user )
|
||||
{
|
||||
return user.find('@') == std::string::npos ;
|
||||
}
|
||||
|
||||
//static
|
||||
bool GSmtp::ProtocolMessage::isValid( const std::string & user )
|
||||
{
|
||||
// only recognise one local mailbox
|
||||
return isPostmaster(user) ;
|
||||
}
|
||||
|
||||
//static
|
||||
bool GSmtp::ProtocolMessage::isPostmaster( std::string user )
|
||||
{
|
||||
G::Str::toUpper( user ) ;
|
||||
G::Str::trim( user , " \t" ) ;
|
||||
return user == "POSTMASTER" ;
|
||||
}
|
||||
|
||||
//static
|
||||
std::string GSmtp::ProtocolMessage::fullName( const std::string & user )
|
||||
{
|
||||
return "Local postmaster <postmaster@localhost>" ;
|
||||
}
|
||||
|
||||
|
@ -27,83 +27,74 @@
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gstrings.h"
|
||||
#include "gverifier.h"
|
||||
#include <string>
|
||||
|
||||
namespace GSmtp
|
||||
{
|
||||
class ProtocolMessage ;
|
||||
class ProtocolMessageImp ;
|
||||
} ;
|
||||
|
||||
// Class: GSmtp::ProtocolMessage
|
||||
// Description: An interface used by the ServerProtocol
|
||||
// class to assemble and process an incoming message.
|
||||
// It implements the three 'buffers' mentioned in
|
||||
// RFC2821 (esp. section 4.1.1). Also does mail-address
|
||||
// validation.
|
||||
// RFC2821 (esp. section 4.1.1).
|
||||
//
|
||||
// This class serves to decouple the ServerProtocol class from
|
||||
// the MessageStore (or whatever else is downstream).
|
||||
// This interface serves to decouple the ServerProtocol class
|
||||
// from the MessageStore (or whatever else is downstream).
|
||||
//
|
||||
class GSmtp::ProtocolMessage
|
||||
{
|
||||
public:
|
||||
ProtocolMessage() ;
|
||||
// Default constructor.
|
||||
class Callback // A callback interface used by ProtocolMessage::process().
|
||||
{
|
||||
public: virtual ~Callback() ;
|
||||
public: virtual void processDone( bool success , unsigned long id , const std::string & reason ) = 0 ;
|
||||
private: void operator=( const Callback & ) ; // not implemented
|
||||
} ;
|
||||
|
||||
~ProtocolMessage() ;
|
||||
virtual ~ProtocolMessage() ;
|
||||
// Destructor.
|
||||
|
||||
static std::pair<bool,std::string> verify( const std::string & ) ;
|
||||
// Checks an address returning
|
||||
// <is-local>|<local-full-name>.
|
||||
//
|
||||
// (If syntactically local then 'first' is
|
||||
// returned true. If local and valid then
|
||||
// 'second' is set to the full description.
|
||||
// If syntactically remote, then 'first'
|
||||
// is returned false and 'second' is empty.)
|
||||
virtual void clear() = 0 ;
|
||||
// Clears the message state and terminates
|
||||
// any asynchronous message processing.
|
||||
|
||||
void clear() ;
|
||||
// Clears the message state.
|
||||
|
||||
bool setFrom( const std::string & from_user ) ;
|
||||
virtual bool setFrom( const std::string & from_user ) = 0 ;
|
||||
// Sets the message envelope 'from'.
|
||||
// Returns false if an invalid user.
|
||||
|
||||
bool addTo( const std::string & to_user ) ;
|
||||
virtual bool addTo( const std::string & to_user , Verifier::Status to_status ) = 0 ;
|
||||
// Adds an envelope 'to'.
|
||||
//
|
||||
// The 'to_status' parameter comes from
|
||||
// GSmtp::Verifier.verify().
|
||||
//
|
||||
// Returns false if an invalid user.
|
||||
// Precondition: setFrom() called
|
||||
// since clear() or process().
|
||||
|
||||
void addReceived( const std::string & ) ;
|
||||
virtual void addReceived( const std::string & ) = 0 ;
|
||||
// Adds a 'received' line to the
|
||||
// start of the content.
|
||||
// Precondition: at least one
|
||||
// successful addTo() call
|
||||
|
||||
void addText( const std::string & ) ;
|
||||
virtual void addText( const std::string & ) = 0 ;
|
||||
// Adds text.
|
||||
// Precondition: at least one
|
||||
// successful addTo() call
|
||||
|
||||
std::string process() ;
|
||||
// Processes and clears the message.
|
||||
// Returns a non-zero-length reason
|
||||
// string on error.
|
||||
virtual void process( Callback & callback ) = 0 ;
|
||||
// Starts asynchronous processing of the
|
||||
// message. Once processing is complete the
|
||||
// message state is cleared and the callback
|
||||
// is triggered. The callback may be called
|
||||
// before process() returns.
|
||||
|
||||
private:
|
||||
static bool isLocal( const std::string & ) ;
|
||||
static bool isValid( const std::string & ) ;
|
||||
static bool isPostmaster( std::string ) ;
|
||||
static std::string fullName( const std::string & ) ;
|
||||
ProtocolMessage( const ProtocolMessage & ) ;
|
||||
void operator=( const ProtocolMessage & ) ;
|
||||
|
||||
private:
|
||||
G::Strings m_to_list ;
|
||||
ProtocolMessageImp * m_imp ;
|
||||
void operator=( const ProtocolMessage & ) ; // not implemented
|
||||
} ;
|
||||
|
||||
#endif
|
||||
|
143
src/main/gprotocolmessageforward.cpp
Normal file
143
src/main/gprotocolmessageforward.cpp
Normal file
@ -0,0 +1,143 @@
|
||||
//
|
||||
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
//
|
||||
// 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, write to the Free Software
|
||||
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gprotocolmessageforward.cpp
|
||||
//
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gprotocolmessageforward.h"
|
||||
#include "gprotocolmessagestore.h"
|
||||
#include "gmessagestore.h"
|
||||
#include "gmemory.h"
|
||||
#include "gstr.h"
|
||||
#include "gassert.h"
|
||||
#include "glog.h"
|
||||
|
||||
GSmtp::ProtocolMessageForward::ProtocolMessageForward( const std::string & server ) :
|
||||
m_server(server) ,
|
||||
m_callback(NULL)
|
||||
{
|
||||
}
|
||||
|
||||
GSmtp::ProtocolMessageForward::~ProtocolMessageForward()
|
||||
{
|
||||
}
|
||||
|
||||
void GSmtp::ProtocolMessageForward::clear()
|
||||
{
|
||||
m_pm.clear() ;
|
||||
m_client <<= 0 ;
|
||||
}
|
||||
|
||||
bool GSmtp::ProtocolMessageForward::setFrom( const std::string & from )
|
||||
{
|
||||
return m_pm.setFrom( from ) ;
|
||||
}
|
||||
|
||||
bool GSmtp::ProtocolMessageForward::addTo( const std::string & to , Verifier::Status to_status )
|
||||
{
|
||||
return m_pm.addTo( to , to_status ) ;
|
||||
}
|
||||
|
||||
void GSmtp::ProtocolMessageForward::addReceived( const std::string & line )
|
||||
{
|
||||
m_pm.addReceived( line ) ;
|
||||
}
|
||||
|
||||
void GSmtp::ProtocolMessageForward::addText( const std::string & line )
|
||||
{
|
||||
m_pm.addText( line ) ;
|
||||
}
|
||||
|
||||
void GSmtp::ProtocolMessageForward::process( ProtocolMessage::Callback & callback )
|
||||
{
|
||||
m_callback = & callback ;
|
||||
m_pm.process( *this ) ;
|
||||
}
|
||||
|
||||
void GSmtp::ProtocolMessageForward::processDone( bool success , unsigned long id , const std::string & reason_in )
|
||||
{
|
||||
std::string reason( reason_in ) ;
|
||||
bool nothing_to_do = false ;
|
||||
if( success )
|
||||
{
|
||||
m_id = id ;
|
||||
success = forward( id , nothing_to_do , &reason ) ;
|
||||
}
|
||||
|
||||
if( nothing_to_do || !success )
|
||||
{
|
||||
G_ASSERT( m_callback != NULL ) ;
|
||||
if( m_callback )
|
||||
m_callback->processDone( success , id , reason ) ;
|
||||
m_callback = NULL ;
|
||||
}
|
||||
}
|
||||
|
||||
bool GSmtp::ProtocolMessageForward::forward( unsigned long id , bool & nothing_to_do , std::string * reason_p )
|
||||
{
|
||||
try
|
||||
{
|
||||
nothing_to_do = false ;
|
||||
*reason_p = std::string() ;
|
||||
G_DEBUG( "GSmtp::ProtocolMessageForward::forward: forwarding message " << id ) ;
|
||||
|
||||
std::auto_ptr<StoredMessage> message = GSmtp::MessageStore::instance().get( id ) ;
|
||||
|
||||
bool ok = true ;
|
||||
if( message->remoteRecipientCount() == 0U )
|
||||
{
|
||||
// use our local delivery mechanism, not the downstream server's
|
||||
nothing_to_do = true ;
|
||||
message->destroy() ; // (already copied to "*.local")
|
||||
}
|
||||
else
|
||||
{
|
||||
m_client <<= new GSmtp::Client( message , *this ) ;
|
||||
std::string reason = m_client->init( m_server ) ;
|
||||
|
||||
ok = reason.empty() ;
|
||||
if( !ok && reason_p != NULL )
|
||||
*reason_p = reason ;
|
||||
}
|
||||
return ok ;
|
||||
}
|
||||
catch( std::exception & e )
|
||||
{
|
||||
if( reason_p != NULL )
|
||||
*reason_p = e.what() ;
|
||||
return false ;
|
||||
}
|
||||
}
|
||||
|
||||
void GSmtp::ProtocolMessageForward::clientDone( std::string reason )
|
||||
{
|
||||
G_DEBUG( "GSmtp::ProtocolMessageForward::clientDone: \"" << reason << "\"" ) ;
|
||||
const bool ok = reason.empty() ;
|
||||
|
||||
G_ASSERT( m_callback != NULL ) ;
|
||||
if( m_callback )
|
||||
{
|
||||
m_callback->processDone( ok , m_id , reason ) ;
|
||||
m_callback = NULL ;
|
||||
}
|
||||
}
|
||||
|
91
src/main/gprotocolmessageforward.h
Normal file
91
src/main/gprotocolmessageforward.h
Normal file
@ -0,0 +1,91 @@
|
||||
//
|
||||
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
//
|
||||
// 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, write to the Free Software
|
||||
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gprotocolmessageforward.h
|
||||
//
|
||||
|
||||
#ifndef G_SMTP_PROTOCOL_MESSAGE_FORWARD_H
|
||||
#define G_SMTP_PROTOCOL_MESSAGE_FORWARD_H
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gprotocolmessage.h"
|
||||
#include "gprotocolmessagestore.h"
|
||||
#include "gsmtpclient.h"
|
||||
#include "gnewmessage.h"
|
||||
#include <string>
|
||||
#include <memory>
|
||||
|
||||
namespace GSmtp
|
||||
{
|
||||
class ProtocolMessageForward ;
|
||||
} ;
|
||||
|
||||
// Class: GSmtp::ProtocolMessageForward
|
||||
// Description: A concrete implementation of the
|
||||
// ProtocolMessage interface which stores incoming
|
||||
// messages in the message store and then immediately
|
||||
// forwards them on to the downstream server.
|
||||
// See also: ProtocolMessageStore
|
||||
//
|
||||
class GSmtp::ProtocolMessageForward : public GSmtp:: ProtocolMessage ,
|
||||
private GSmtp:: ProtocolMessage::Callback ,
|
||||
private GSmtp:: Client::ClientCallback
|
||||
{
|
||||
public:
|
||||
explicit ProtocolMessageForward( const std::string & server_address ) ;
|
||||
// Constructor.
|
||||
|
||||
virtual ~ProtocolMessageForward() ;
|
||||
// Destructor.
|
||||
|
||||
virtual void clear() ;
|
||||
// See ProtocolMessage.
|
||||
|
||||
virtual bool setFrom( const std::string & from_user ) ;
|
||||
// See ProtocolMessage.
|
||||
|
||||
virtual bool addTo( const std::string & to_user , Verifier::Status to_status ) ;
|
||||
// See ProtocolMessage.
|
||||
|
||||
virtual void addReceived( const std::string & ) ;
|
||||
// See ProtocolMessage.
|
||||
|
||||
virtual void addText( const std::string & ) ;
|
||||
// See ProtocolMessage.
|
||||
|
||||
virtual void process( ProtocolMessage::Callback & callback ) ;
|
||||
// See ProtocolMessage.
|
||||
|
||||
private:
|
||||
void operator=( const ProtocolMessageForward & ) ; // not implemented
|
||||
virtual void processDone( bool , unsigned long , const std::string & ) ; // from ProtocolMessage::Callback
|
||||
virtual void clientDone( std::string ) ; // from Client::ClientCallback
|
||||
bool forward( unsigned long , bool & , std::string * ) ;
|
||||
|
||||
private:
|
||||
ProtocolMessageStore m_pm ;
|
||||
std::string m_server ;
|
||||
ProtocolMessage::Callback * m_callback ;
|
||||
std::auto_ptr<Client> m_client ;
|
||||
unsigned long m_id ;
|
||||
} ;
|
||||
|
||||
#endif
|
126
src/main/gprotocolmessagestore.cpp
Normal file
126
src/main/gprotocolmessagestore.cpp
Normal file
@ -0,0 +1,126 @@
|
||||
//
|
||||
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
//
|
||||
// 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, write to the Free Software
|
||||
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gprotocolmessagestore.cpp
|
||||
//
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gprotocolmessagestore.h"
|
||||
#include "gmessagestore.h"
|
||||
#include "gmemory.h"
|
||||
#include "gstr.h"
|
||||
#include "gassert.h"
|
||||
#include "glog.h"
|
||||
|
||||
GSmtp::ProtocolMessageStore::ProtocolMessageStore()
|
||||
{
|
||||
}
|
||||
|
||||
GSmtp::ProtocolMessageStore::~ProtocolMessageStore()
|
||||
{
|
||||
}
|
||||
|
||||
void GSmtp::ProtocolMessageStore::clear()
|
||||
{
|
||||
m_msg <<= 0 ;
|
||||
}
|
||||
|
||||
bool GSmtp::ProtocolMessageStore::setFrom( const std::string & from )
|
||||
{
|
||||
try
|
||||
{
|
||||
if( from.length() == 0U )
|
||||
return false ;
|
||||
|
||||
G_ASSERT( m_msg.get() == NULL ) ;
|
||||
clear() ; // just in case
|
||||
|
||||
std::auto_ptr<NewMessage> new_message( MessageStore::instance().newMessage(from) ) ;
|
||||
m_msg <<= new_message.release() ;
|
||||
|
||||
return true ;
|
||||
}
|
||||
catch( std::exception & e )
|
||||
{
|
||||
G_ERROR( "GSmtp::ProtocolMessage::setFrom: error: " << e.what() ) ;
|
||||
return false ;
|
||||
}
|
||||
}
|
||||
|
||||
bool GSmtp::ProtocolMessageStore::addTo( const std::string & to , Verifier::Status to_status )
|
||||
{
|
||||
G_ASSERT( m_msg.get() != NULL ) ;
|
||||
if( to.length() > 0U && m_msg.get() != NULL )
|
||||
{
|
||||
const bool is_local = to_status.first ;
|
||||
const bool is_valid = is_local && to_status.second.length() != 0U ;
|
||||
if( is_local && !is_valid )
|
||||
{
|
||||
G_WARNING( "GSmtp::ProtocolMessage: rejecting local recipent (not postmaster): " << to ) ;
|
||||
return false ;
|
||||
}
|
||||
else
|
||||
{
|
||||
m_msg->addTo( to , is_local ) ;
|
||||
return true ;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
return false ;
|
||||
}
|
||||
}
|
||||
|
||||
void GSmtp::ProtocolMessageStore::addReceived( const std::string & line )
|
||||
{
|
||||
addText( line ) ;
|
||||
}
|
||||
|
||||
void GSmtp::ProtocolMessageStore::addText( const std::string & line )
|
||||
{
|
||||
G_ASSERT( m_msg.get() != NULL ) ;
|
||||
if( m_msg.get() != NULL )
|
||||
m_msg->addText( line ) ;
|
||||
}
|
||||
|
||||
void GSmtp::ProtocolMessageStore::process( Callback & callback )
|
||||
{
|
||||
try
|
||||
{
|
||||
G_ASSERT( m_msg.get() != NULL ) ;
|
||||
unsigned long id = 0UL ;
|
||||
if( m_msg.get() != NULL )
|
||||
{
|
||||
m_msg->store() ;
|
||||
id = m_msg->id() ;
|
||||
}
|
||||
clear() ;
|
||||
callback.processDone( true , id , std::string() ) ;
|
||||
}
|
||||
catch( std::exception & e )
|
||||
{
|
||||
G_ERROR( "GSmtp::ProtocolMessage::process: error: " << e.what() ) ;
|
||||
clear() ;
|
||||
callback.processDone( false , 0UL , e.what() ) ;
|
||||
}
|
||||
}
|
||||
|
||||
|
80
src/main/gprotocolmessagestore.h
Normal file
80
src/main/gprotocolmessagestore.h
Normal file
@ -0,0 +1,80 @@
|
||||
//
|
||||
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
//
|
||||
// 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, write to the Free Software
|
||||
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gprotocolmessagestore.h
|
||||
//
|
||||
|
||||
#ifndef G_SMTP_PROTOCOL_MESSAGE_STORE_H
|
||||
#define G_SMTP_PROTOCOL_MESSAGE_STORE_H
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gprotocolmessage.h"
|
||||
#include "gnewmessage.h"
|
||||
#include <string>
|
||||
#include <memory>
|
||||
|
||||
namespace GSmtp
|
||||
{
|
||||
class ProtocolMessageStore ;
|
||||
} ;
|
||||
|
||||
// Class: GSmtp::ProtocolMessageStore
|
||||
// Description: A concrete implementation of the
|
||||
// ProtocolMessage interface which stores incoming
|
||||
// messages in the message store.
|
||||
// See also: ProtocolMessageForward
|
||||
//
|
||||
class GSmtp::ProtocolMessageStore : public GSmtp:: ProtocolMessage
|
||||
{
|
||||
public:
|
||||
ProtocolMessageStore() ;
|
||||
// Constructor.
|
||||
|
||||
virtual ~ProtocolMessageStore() ;
|
||||
// Destructor.
|
||||
|
||||
virtual void clear() ;
|
||||
// See ProtocolMessage.
|
||||
|
||||
virtual bool setFrom( const std::string & from_user ) ;
|
||||
// See ProtocolMessage.
|
||||
|
||||
virtual bool addTo( const std::string & to_user , Verifier::Status to_status ) ;
|
||||
// See ProtocolMessage.
|
||||
|
||||
virtual void addReceived( const std::string & ) ;
|
||||
// See ProtocolMessage.
|
||||
|
||||
virtual void addText( const std::string & ) ;
|
||||
// See ProtocolMessage.
|
||||
|
||||
virtual void process( ProtocolMessage::Callback & callback ) ;
|
||||
// See ProtocolMessage.
|
||||
|
||||
private:
|
||||
void operator=( const ProtocolMessageStore & ) ; // not implemented
|
||||
|
||||
private:
|
||||
std::auto_ptr<NewMessage> m_msg ;
|
||||
} ;
|
||||
|
||||
#endif
|
||||
|
@ -32,12 +32,13 @@
|
||||
#include "gassert.h"
|
||||
#include <string>
|
||||
|
||||
GSmtp::ServerProtocol::ServerProtocol( Sender & sender , const std::string & thishost ,
|
||||
const std::string & peer_address ) :
|
||||
GSmtp::ServerProtocol::ServerProtocol( Sender & sender , Verifier & verifier , ProtocolMessage & pmessage ,
|
||||
const std::string & thishost , const std::string & peer_address ) :
|
||||
m_thishost(thishost) ,
|
||||
m_sender(sender) ,
|
||||
m_verifier(verifier) ,
|
||||
m_pmessage(pmessage) ,
|
||||
m_state(sStart) ,
|
||||
m_ss(NULL) ,
|
||||
m_peer_address(peer_address)
|
||||
{
|
||||
// (dont send anything to the peer from this ctor -- the Sender
|
||||
@ -71,7 +72,9 @@ void GSmtp::ServerProtocol::addTransition( Event e , State state_from , State st
|
||||
|
||||
void GSmtp::ServerProtocol::sendGreeting( const std::string & thishost , const std::string & ident )
|
||||
{
|
||||
ss() << "220 " << thishost << " -- " << ident << " -- Service ready" << end() ;
|
||||
std::stringstream ss ;
|
||||
ss << "220 " << thishost << " -- " << ident << " -- Service ready" ;
|
||||
send( ss.str() ) ;
|
||||
}
|
||||
|
||||
bool GSmtp::ServerProtocol::apply( const std::string & line )
|
||||
@ -82,26 +85,34 @@ bool GSmtp::ServerProtocol::apply( const std::string & line )
|
||||
{
|
||||
G_LOG( "GSmtp::ServerProtocol: rx<<: [message content not logged]" ) ;
|
||||
G_LOG( "GSmtp::ServerProtocol: rx<<: \"" << G::Str::toPrintableAscii(line) << "\"" ) ;
|
||||
std::string reason = m_message.process() ;
|
||||
const bool success = reason.empty() ;
|
||||
m_state = sIdle ;
|
||||
sendCompletionReply( success , reason ) ;
|
||||
m_state = sProcessing ;
|
||||
m_pmessage.process( *this ) ; // processDone() callback
|
||||
}
|
||||
else
|
||||
{
|
||||
m_message.addText( isEscaped(line) ? line.substr(1U) : line ) ;
|
||||
m_pmessage.addText( isEscaped(line) ? line.substr(1U) : line ) ;
|
||||
}
|
||||
return false ;
|
||||
}
|
||||
else
|
||||
{
|
||||
G_LOG( "GSmtp::ServerProtocol: rx<<: \"" << G::Str::toPrintableAscii(line) << "\"" ) ;
|
||||
Event event = commandEvent( commandString(line) ) ;
|
||||
State new_state = applyEvent( event , line ) ;
|
||||
Event event = commandEvent( commandWord(line) ) ;
|
||||
State new_state = applyEvent( event , commandLine(line) ) ;
|
||||
return new_state == sEnd ;
|
||||
}
|
||||
}
|
||||
|
||||
void GSmtp::ServerProtocol::processDone( bool success , unsigned long , const std::string & reason )
|
||||
{
|
||||
G_ASSERT( m_state == sProcessing ) ; // (a RSET will call m_pmessage.clear() to cancel the callback)
|
||||
if( m_state == sProcessing ) // just in case
|
||||
{
|
||||
m_state = sIdle ;
|
||||
sendCompletionReply( success , reason ) ;
|
||||
}
|
||||
}
|
||||
|
||||
GSmtp::ServerProtocol::State GSmtp::ServerProtocol::applyEvent( Event event , const std::string & line )
|
||||
{
|
||||
// look up in the multimap keyed on current-state + event
|
||||
@ -154,17 +165,26 @@ void GSmtp::ServerProtocol::doNoop( const std::string & , bool & )
|
||||
|
||||
void GSmtp::ServerProtocol::doVrfy( const std::string & line , bool & )
|
||||
{
|
||||
size_t pos = line.find_first_of( " \t" ) ;
|
||||
std::string user = line.substr(pos) ;
|
||||
G::Str::trimLeft( user , " \t" ) ;
|
||||
std::pair<bool,std::string> rc = ProtocolMessage::verify( user ) ;
|
||||
std::string mbox = parseMailbox( line ) ;
|
||||
Verifier::Status rc = m_verifier.verify( mbox ) ;
|
||||
bool local = rc.first ;
|
||||
if( local && rc.second.length() )
|
||||
sendVerified( rc.second ) ;
|
||||
else if( local )
|
||||
sendNotVerified( rc.second ) ;
|
||||
sendNotVerified( mbox ) ;
|
||||
else
|
||||
sendWillAccept( rc.second ) ;
|
||||
sendWillAccept( mbox ) ;
|
||||
}
|
||||
|
||||
std::string GSmtp::ServerProtocol::parseMailbox( const std::string & line ) const
|
||||
{
|
||||
std::string user ;
|
||||
size_t pos = line.find_first_of( " \t" ) ;
|
||||
if( pos != std::string::npos )
|
||||
user = line.substr(pos) ;
|
||||
|
||||
G::Str::trim( user , " \t" ) ;
|
||||
return user ;
|
||||
}
|
||||
|
||||
void GSmtp::ServerProtocol::doEhlo( const std::string & line , bool & predicate )
|
||||
@ -178,7 +198,7 @@ void GSmtp::ServerProtocol::doEhlo( const std::string & line , bool & predicate
|
||||
else
|
||||
{
|
||||
m_peer_name = peer_name ;
|
||||
m_message.clear() ;
|
||||
m_pmessage.clear() ;
|
||||
sendEhloReply( m_thishost ) ;
|
||||
}
|
||||
}
|
||||
@ -194,16 +214,16 @@ void GSmtp::ServerProtocol::doHelo( const std::string & line , bool & predicate
|
||||
else
|
||||
{
|
||||
m_peer_name = peer_name ;
|
||||
m_message.clear() ;
|
||||
m_pmessage.clear() ;
|
||||
sendHeloReply( m_thishost ) ;
|
||||
}
|
||||
}
|
||||
|
||||
void GSmtp::ServerProtocol::doMail( const std::string & line , bool & predicate )
|
||||
{
|
||||
m_message.clear() ;
|
||||
m_pmessage.clear() ;
|
||||
std::string from = parseFrom( line ) ;
|
||||
bool ok = m_message.setFrom( from ) ;
|
||||
bool ok = m_pmessage.setFrom( from ) ;
|
||||
predicate = ok ;
|
||||
if( ok )
|
||||
sendMailReply() ;
|
||||
@ -214,7 +234,7 @@ void GSmtp::ServerProtocol::doMail( const std::string & line , bool & predicate
|
||||
void GSmtp::ServerProtocol::doRcpt( const std::string & line , bool & predicate )
|
||||
{
|
||||
std::string to = parseTo( line ) ;
|
||||
bool ok = m_message.addTo( to ) ;
|
||||
bool ok = m_pmessage.addTo( to , m_verifier.verify(to) ) ;
|
||||
predicate = ok ;
|
||||
if( ok )
|
||||
sendRcptReply() ;
|
||||
@ -229,7 +249,7 @@ void GSmtp::ServerProtocol::doUnknown( const std::string & line , bool & )
|
||||
|
||||
void GSmtp::ServerProtocol::doRset( const std::string & line , bool & )
|
||||
{
|
||||
m_message.clear() ;
|
||||
m_pmessage.clear() ;
|
||||
sendRsetReply() ;
|
||||
}
|
||||
|
||||
@ -240,7 +260,7 @@ void GSmtp::ServerProtocol::doNoRecipients( const std::string & line , bool & )
|
||||
|
||||
void GSmtp::ServerProtocol::doData( const std::string & line , bool & )
|
||||
{
|
||||
m_message.addReceived( receivedLine() ) ;
|
||||
m_pmessage.addReceived( receivedLine() ) ;
|
||||
sendDataReply() ;
|
||||
}
|
||||
|
||||
@ -264,14 +284,25 @@ bool GSmtp::ServerProtocol::isEscaped( const std::string & line ) const
|
||||
return line.length() > 1U && line.at(0U) == '.' ;
|
||||
}
|
||||
|
||||
std::string GSmtp::ServerProtocol::commandString( const std::string & line ) const
|
||||
std::string GSmtp::ServerProtocol::commandWord( const std::string & line_in ) const
|
||||
{
|
||||
size_t ws_pos = line.find_first_of( " \t" ) ;
|
||||
std::string command = line.substr( 0U , ws_pos ) ;
|
||||
std::string line( line_in ) ;
|
||||
G::Str::trimLeft( line , " \t" ) ;
|
||||
|
||||
size_t pos = line.find_first_of( " \t" ) ;
|
||||
std::string command = line.substr( 0U , pos ) ;
|
||||
|
||||
G::Str::toUpper( command ) ;
|
||||
return command ;
|
||||
}
|
||||
|
||||
std::string GSmtp::ServerProtocol::commandLine( const std::string & line_in ) const
|
||||
{
|
||||
std::string line( line_in ) ;
|
||||
G::Str::trimLeft( line , " \t" ) ;
|
||||
return line ;
|
||||
}
|
||||
|
||||
//static
|
||||
GSmtp::ServerProtocol::Event GSmtp::ServerProtocol::commandEvent( const std::string & command )
|
||||
{
|
||||
@ -358,11 +389,12 @@ void GSmtp::ServerProtocol::sendBadTo( const std::string & to )
|
||||
|
||||
void GSmtp::ServerProtocol::sendEhloReply( const std::string & domain )
|
||||
{
|
||||
ss()
|
||||
std::stringstream ss ;
|
||||
ss
|
||||
<< "250-" << domain << " says hello" << crlf()
|
||||
//<<"250-XYZEXTENSION" << crlf()
|
||||
<< "250 8BITMIME"
|
||||
<< end() ;
|
||||
<< "250 8BITMIME" ;
|
||||
send( ss.str() ) ;
|
||||
}
|
||||
|
||||
void GSmtp::ServerProtocol::sendHeloReply( const std::string & domain )
|
||||
@ -375,25 +407,6 @@ void GSmtp::ServerProtocol::sendOk()
|
||||
send( "250 OK" ) ;
|
||||
}
|
||||
|
||||
std::ostream & GSmtp::ServerProtocol::ss()
|
||||
{
|
||||
delete m_ss ;
|
||||
m_ss = NULL ;
|
||||
m_ss = new std::stringstream ;
|
||||
return *m_ss ;
|
||||
}
|
||||
|
||||
GSmtp::ServerProtocol::End GSmtp::ServerProtocol::end()
|
||||
{
|
||||
return End(*this) ;
|
||||
}
|
||||
|
||||
void GSmtp::ServerProtocol::onEnd()
|
||||
{
|
||||
G_ASSERT( m_ss != NULL ) ;
|
||||
send( m_ss->str() ) ;
|
||||
}
|
||||
|
||||
//static
|
||||
std::string GSmtp::ServerProtocol::crlf()
|
||||
{
|
||||
@ -409,7 +422,6 @@ void GSmtp::ServerProtocol::send( std::string line )
|
||||
|
||||
GSmtp::ServerProtocol::~ServerProtocol()
|
||||
{
|
||||
delete m_ss ;
|
||||
}
|
||||
|
||||
std::string GSmtp::ServerProtocol::parseFrom( const std::string & line ) const
|
||||
@ -454,7 +466,7 @@ std::string GSmtp::ServerProtocol::parsePeerName( const std::string & line ) con
|
||||
return std::string() ;
|
||||
|
||||
std::string peer_name = line.substr( pos + 1U ) ;
|
||||
G::Str::trimLeft( peer_name , " \t" ) ;
|
||||
G::Str::trim( peer_name , " \t" ) ;
|
||||
return peer_name ;
|
||||
}
|
||||
|
||||
@ -491,13 +503,6 @@ GSmtp::ServerProtocol::Sender::~Sender()
|
||||
|
||||
// ===
|
||||
|
||||
GSmtp::ServerProtocol::End::End( ServerProtocol & p ) :
|
||||
m_p(p)
|
||||
{
|
||||
}
|
||||
|
||||
// ===
|
||||
|
||||
GSmtp::ServerProtocol::Transition::Transition( State s1 , State s2 , Action a , State s3 ) :
|
||||
from(s1) ,
|
||||
to(s2) ,
|
||||
|
@ -27,6 +27,7 @@
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gprotocolmessage.h"
|
||||
#include "gverifier.h"
|
||||
#include <map>
|
||||
|
||||
namespace GSmtp
|
||||
@ -56,36 +57,40 @@ namespace GSmtp
|
||||
//
|
||||
// See also: ProtocolMessage, RFC2821
|
||||
//
|
||||
class GSmtp::ServerProtocol
|
||||
class GSmtp::ServerProtocol : private GSmtp:: ProtocolMessage::Callback
|
||||
{
|
||||
public:
|
||||
class Sender // Used to send protocol replies.
|
||||
class Sender // An interface used by ServerProtocol to send protocol replies.
|
||||
{
|
||||
public: virtual void protocolSend( const std::string & s ) = 0 ;
|
||||
public: virtual void protocolDone() = 0 ;
|
||||
public: virtual ~Sender() ;
|
||||
private: void operator=( const Sender & ) ;
|
||||
private: void operator=( const Sender & ) ; // not implemented
|
||||
} ;
|
||||
|
||||
class End // A private implementation class.
|
||||
{
|
||||
public: ServerProtocol & m_p ;
|
||||
public: End( ServerProtocol & p ) ;
|
||||
} ;
|
||||
|
||||
ServerProtocol( Sender & sender , const std::string & thishost ,
|
||||
const std::string & peer_address ) ;
|
||||
// Constructor. The Sender interface is used to
|
||||
// send protocol replies back to the client.
|
||||
// The 'thishost' string is used in HELO
|
||||
// replies (etc.) The peer address string is
|
||||
// put into the "Received:" trace lines.
|
||||
ServerProtocol( Sender & sender , Verifier & verifier , ProtocolMessage & pmessage ,
|
||||
const std::string & thishost , const std::string & peer_address ) ;
|
||||
// Constructor.
|
||||
//
|
||||
// The Verifier interface is used to verify recipient
|
||||
// addresses. See GSmtp::Verifier.
|
||||
//
|
||||
// The ProtocolMessage interface is used to assemble and
|
||||
// process an incoming message.
|
||||
//
|
||||
// The Sender interface is used to send protocol
|
||||
// replies back to the client.
|
||||
//
|
||||
// The 'thishost' string is used in HELO replies (etc).
|
||||
//
|
||||
// The peer address string is put into the "Received:"
|
||||
// trace lines.
|
||||
|
||||
void init( const std::string & ident ) ;
|
||||
// Starts the protocol. The 'ident' string is issued
|
||||
// to the client.
|
||||
|
||||
~ServerProtocol() ;
|
||||
virtual ~ServerProtocol() ;
|
||||
// Destructor.
|
||||
|
||||
bool apply( const std::string & line ) ;
|
||||
@ -94,9 +99,6 @@ public:
|
||||
// Returns true if the protocol has completed
|
||||
// and Sender::protocolDone() has been called.
|
||||
|
||||
void onEnd() ;
|
||||
// A pseudo-private method used by the End class.
|
||||
|
||||
private:
|
||||
enum Event
|
||||
{
|
||||
@ -120,6 +122,7 @@ private:
|
||||
sGotMail ,
|
||||
sGotRcpt ,
|
||||
sData ,
|
||||
sProcessing ,
|
||||
s_Any ,
|
||||
s_Same
|
||||
} ;
|
||||
@ -135,15 +138,15 @@ private:
|
||||
typedef std::multimap<Event,Transition GLessAllocator(Event,Transition) > Map ;
|
||||
|
||||
private:
|
||||
ServerProtocol( const ServerProtocol & ) ;
|
||||
void operator=( const ServerProtocol & ) ;
|
||||
ServerProtocol( const ServerProtocol & ) ; // not implemented
|
||||
void operator=( const ServerProtocol & ) ; // not implemented
|
||||
State applyEvent( Event , const std::string & ) ;
|
||||
void send( std::string ) ;
|
||||
static Event commandEvent( const std::string & ) ;
|
||||
std::string commandString( const std::string & line ) const ;
|
||||
std::ostream & ss() ;
|
||||
End end() ;
|
||||
std::string commandWord( const std::string & line ) const ;
|
||||
std::string commandLine( const std::string & line ) const ;
|
||||
static std::string crlf() ;
|
||||
virtual void processDone( bool , unsigned long , const std::string & ) ; // from ProtocolMessage
|
||||
bool isEndOfText( const std::string & ) const ;
|
||||
bool isEscaped( const std::string & ) const ;
|
||||
void addTransition( Event , State old , State new_ , Action , State alt = s_Same ) ;
|
||||
@ -179,29 +182,20 @@ private:
|
||||
void sendOk() ;
|
||||
std::string parseFrom( const std::string & ) const ;
|
||||
std::string parseTo( const std::string & ) const ;
|
||||
std::string parseMailbox( const std::string & ) const ;
|
||||
std::string parsePeerName( const std::string & ) const ;
|
||||
std::string parse( const std::string & ) const ;
|
||||
std::string receivedLine() const ;
|
||||
|
||||
private:
|
||||
Sender & m_sender ;
|
||||
ProtocolMessage & m_pmessage ;
|
||||
Verifier & m_verifier ;
|
||||
State m_state ;
|
||||
Map m_map ;
|
||||
ProtocolMessage m_message ;
|
||||
std::stringstream * m_ss ;
|
||||
std::string m_thishost ;
|
||||
std::string m_peer_name ;
|
||||
std::string m_peer_address ;
|
||||
} ;
|
||||
|
||||
namespace GSmtp
|
||||
{
|
||||
inline
|
||||
std::ostream & operator<<( std::ostream & stream , const ServerProtocol::End & e )
|
||||
{
|
||||
e.m_p.onEnd() ;
|
||||
return stream ;
|
||||
}
|
||||
} ;
|
||||
|
||||
#endif
|
||||
|
@ -30,6 +30,7 @@
|
||||
#include "gmemory.h"
|
||||
#include "gsmtpclient.h"
|
||||
#include "gresolve.h"
|
||||
#include "gassert.h"
|
||||
#include "glog.h"
|
||||
|
||||
//static
|
||||
@ -38,20 +39,31 @@ std::string GSmtp::Client::crlf()
|
||||
return std::string("\015\012") ;
|
||||
}
|
||||
|
||||
GSmtp::Client::Client( GSmtp::MessageStore & store , bool quit_on_disconnect ) :
|
||||
GSmtp::Client::Client( MessageStore & store , bool quit_on_disconnect ) :
|
||||
GNet::Client(false,quit_on_disconnect) ,
|
||||
m_callback(NULL) ,
|
||||
m_store(store) ,
|
||||
m_store(&store) ,
|
||||
m_buffer(crlf()) ,
|
||||
m_protocol(*this,GNet::Local::fqdn()) ,
|
||||
m_socket(NULL)
|
||||
{
|
||||
}
|
||||
|
||||
GSmtp::Client::Client( GSmtp::MessageStore & store , ClientCallback & callback , bool quit_on_disconnect ) :
|
||||
GSmtp::Client::Client( MessageStore & store , ClientCallback & callback , bool quit_on_disconnect ) :
|
||||
GNet::Client(false,quit_on_disconnect) ,
|
||||
m_callback(&callback) ,
|
||||
m_store(store) ,
|
||||
m_store(&store) ,
|
||||
m_buffer(crlf()) ,
|
||||
m_protocol(*this,GNet::Local::fqdn()) ,
|
||||
m_socket(NULL)
|
||||
{
|
||||
}
|
||||
|
||||
GSmtp::Client::Client( std::auto_ptr<StoredMessage> message , ClientCallback & callback ) :
|
||||
GNet::Client(false,false) ,
|
||||
m_callback(&callback) ,
|
||||
m_store(NULL) ,
|
||||
m_message(message) ,
|
||||
m_buffer(crlf()) ,
|
||||
m_protocol(*this,GNet::Local::fqdn()) ,
|
||||
m_socket(NULL)
|
||||
@ -69,7 +81,7 @@ std::string GSmtp::Client::init( const std::string & s )
|
||||
|
||||
std::string GSmtp::Client::init( const std::string & host , const std::string & service )
|
||||
{
|
||||
if( m_store.empty() )
|
||||
if( m_store != NULL && m_store->empty() )
|
||||
return "no messages to send" ;
|
||||
|
||||
std::string error ;
|
||||
@ -107,20 +119,17 @@ bool GSmtp::Client::protocolSend( const std::string & line )
|
||||
void GSmtp::Client::onConnect( GNet::Socket & socket )
|
||||
{
|
||||
m_socket = &socket ;
|
||||
m_iter = m_store.iterator() ;
|
||||
if( m_store != NULL )
|
||||
{
|
||||
m_iter = m_store->iterator() ;
|
||||
if( !sendNext() )
|
||||
finish() ;
|
||||
}
|
||||
|
||||
void GSmtp::Client::finish()
|
||||
{
|
||||
if( m_callback != NULL )
|
||||
{
|
||||
m_callback->onCompletion(std::string()) ;
|
||||
m_callback = NULL ;
|
||||
}
|
||||
|
||||
disconnect() ; // GNet::Client::disconnect()
|
||||
else
|
||||
{
|
||||
G_ASSERT( m_message.get() != NULL ) ;
|
||||
start( *m_message.get() ) ;
|
||||
}
|
||||
}
|
||||
|
||||
bool GSmtp::Client::sendNext()
|
||||
@ -140,29 +149,41 @@ bool GSmtp::Client::sendNext()
|
||||
m_message = message ;
|
||||
}
|
||||
|
||||
m_protocol.start( m_message->from() , m_message->to() , m_message->eightBit() ,
|
||||
m_message->extractContentStream() , *this ) ;
|
||||
start( *m_message.get() ) ;
|
||||
return true ;
|
||||
}
|
||||
|
||||
void GSmtp::Client::callback( bool ok )
|
||||
void GSmtp::Client::start( StoredMessage & message )
|
||||
{
|
||||
G_DEBUG( "GSmtp::Client::callback: " << ok ) ;
|
||||
m_protocol.start( message.from() , message.to() , message.eightBit() ,
|
||||
message.extractContentStream() , *this ) ;
|
||||
}
|
||||
|
||||
void GSmtp::Client::protocolDone( bool ok , const std::string & reason )
|
||||
{
|
||||
G_DEBUG( "GSmtp::Client::protocolDone: " << ok << ": " << reason ) ;
|
||||
|
||||
std::string error_message ;
|
||||
if( !ok )
|
||||
error_message = std::string("smtp client protocol failure: ") + reason ;
|
||||
|
||||
if( m_message.get() != NULL )
|
||||
{
|
||||
if( ok )
|
||||
m_message->destroy() ;
|
||||
else
|
||||
m_message->fail("smtp protocol failure") ;
|
||||
m_message->fail( error_message ) ;
|
||||
}
|
||||
|
||||
if( m_store == NULL || !sendNext() )
|
||||
{
|
||||
finish( error_message ) ;
|
||||
}
|
||||
if( !sendNext() )
|
||||
finish() ;
|
||||
}
|
||||
|
||||
void GSmtp::Client::onDisconnect()
|
||||
{
|
||||
if( m_callback != NULL )
|
||||
m_callback->onCompletion( "connection to server lost" ) ;
|
||||
doCallback( "connection to server lost" ) ;
|
||||
m_socket = NULL ;
|
||||
}
|
||||
|
||||
@ -184,25 +205,38 @@ void GSmtp::Client::onData( const char * data , size_t size )
|
||||
}
|
||||
}
|
||||
|
||||
void GSmtp::Client::onError( const std::string &error )
|
||||
void GSmtp::Client::onError( const std::string & error )
|
||||
{
|
||||
doCallback( std::string("error on connection to server: ") + error ) ;
|
||||
G_WARNING( "GSmtp::Client: error: \"" << error << "\"" ) ;
|
||||
}
|
||||
|
||||
void GSmtp::Client::finish( const std::string & reason )
|
||||
{
|
||||
doCallback( reason ) ;
|
||||
disconnect() ; // GNet::Client::disconnect()
|
||||
}
|
||||
|
||||
void GSmtp::Client::doCallback( const std::string & reason )
|
||||
{
|
||||
if( m_callback != NULL )
|
||||
m_callback->onCompletion( std::string("error on connection to server: ") + error ) ;
|
||||
G_WARNING( "GSmtp::Client: error: \"" << error << "\"" ) ;
|
||||
{
|
||||
m_callback->clientDone( reason ) ;
|
||||
m_callback = NULL ;
|
||||
}
|
||||
}
|
||||
|
||||
void GSmtp::Client::onWriteable()
|
||||
{
|
||||
if( protocolSend(m_pending) )
|
||||
{
|
||||
m_protocol.sendComplete() ;
|
||||
m_protocol.sendDone() ;
|
||||
}
|
||||
}
|
||||
|
||||
// ===
|
||||
|
||||
void GSmtp::Client::ClientCallback::onCompletion( std::string )
|
||||
GSmtp::Client::ClientCallback::~ClientCallback()
|
||||
{
|
||||
// empty
|
||||
}
|
||||
|
||||
|
@ -31,6 +31,7 @@
|
||||
#include "gclient.h"
|
||||
#include "gclientprotocol.h"
|
||||
#include "gmessagestore.h"
|
||||
#include "gstoredmessage.h"
|
||||
#include "gsocket.h"
|
||||
#include "gstrings.h"
|
||||
#include "gexception.h"
|
||||
@ -49,17 +50,22 @@ namespace GSmtp
|
||||
// a remote SMTP server.
|
||||
//
|
||||
class GSmtp::Client : private GNet::Client ,
|
||||
private GSmtp::ClientProtocol::Sender , private GSmtp::ClientProtocol::Callback
|
||||
private GSmtp:: ClientProtocol::Sender , private GSmtp:: ClientProtocol::Callback
|
||||
{
|
||||
public:
|
||||
G_EXCEPTION( NotConnected , "not connected" ) ;
|
||||
class ClientCallback // A callback interface used by GSmtp::Client.
|
||||
{
|
||||
public: virtual void onCompletion( std::string ) ;
|
||||
public: virtual void clientDone( std::string ) = 0 ;
|
||||
public: virtual ~ClientCallback() ;
|
||||
private: void operator=( const ClientCallback & ) ; // not implemented
|
||||
} ;
|
||||
|
||||
Client( MessageStore & store , bool quit_on_disconnect ) ;
|
||||
// Constructor. The reference is kept.
|
||||
// Constructor. The message-store reference is kept.
|
||||
//
|
||||
// The 'quit_on_disconnect' parameter refers to
|
||||
// GNet::EventSources::quit().
|
||||
|
||||
Client( MessageStore & store , ClientCallback & callback , bool quit_on_disconnect ) ;
|
||||
// Constructor. The references are kept.
|
||||
@ -69,20 +75,27 @@ public:
|
||||
// or that the server connection has
|
||||
// been lost.
|
||||
|
||||
std::string init( const std::string & host , const std::string & service ) ;
|
||||
Client( std::auto_ptr<StoredMessage> message , ClientCallback & callback ) ;
|
||||
// Constructor for sending a single message.
|
||||
//
|
||||
// The callback is used to signal that
|
||||
// all message processing has finished
|
||||
// or that the server connection has
|
||||
// been lost.
|
||||
|
||||
std::string init( const std::string & server_address_string ) ;
|
||||
// Starts the sending process. Messages
|
||||
// are extracted from the message store
|
||||
// (as passed in the ctor) and forwarded
|
||||
// on to the specified server.
|
||||
//
|
||||
// To be called once (only) after construction.
|
||||
//
|
||||
// Returns an error string if there are no messages
|
||||
// to be sent, or if the network connection
|
||||
// cannot be initiated. Returns the empty
|
||||
// string on success.
|
||||
|
||||
std::string init( const std::string & host_service ) ;
|
||||
// An overload.
|
||||
|
||||
bool busy() const ;
|
||||
// Returns true if the client is still
|
||||
// busy processing messages.
|
||||
@ -93,15 +106,18 @@ private:
|
||||
virtual void onData( const char * data , size_t size ) ; // GNet::Client
|
||||
virtual void onWriteable() ; // GNet::Client
|
||||
virtual void onError( const std::string & error ) ; // GNet::Client
|
||||
virtual bool protocolSend( const std::string & ) ; // Sender
|
||||
virtual void callback( bool ) ; // ClientCallback
|
||||
virtual bool protocolSend( const std::string & ) ; // ClientProtocol::Sender
|
||||
virtual void protocolDone( bool , const std::string & ) ; // ClientProtocol::Callback
|
||||
std::string init( const std::string & , const std::string & ) ;
|
||||
GNet::Socket & socket() ;
|
||||
static std::string crlf() ;
|
||||
bool sendNext() ;
|
||||
void finish() ;
|
||||
void start( StoredMessage & ) ;
|
||||
void doCallback( const std::string & ) ;
|
||||
void finish( const std::string & reason = std::string() ) ;
|
||||
|
||||
private:
|
||||
MessageStore & m_store ;
|
||||
MessageStore * m_store ;
|
||||
std::auto_ptr<StoredMessage> m_message ;
|
||||
MessageStore::Iterator m_iter ;
|
||||
GNet::LineBuffer m_buffer ;
|
||||
|
@ -24,6 +24,9 @@
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gsmtpserver.h"
|
||||
#include "gprotocolmessagestore.h"
|
||||
#include "gprotocolmessageforward.h"
|
||||
#include "gmemory.h"
|
||||
#include "glocal.h"
|
||||
#include "glog.h"
|
||||
#include "gdebug.h"
|
||||
@ -31,9 +34,10 @@
|
||||
#include <string>
|
||||
|
||||
GSmtp::ServerPeer::ServerPeer( GNet::StreamSocket * socket , GNet::Address peer_address ,
|
||||
Server & server , const std::string & ident ) :
|
||||
Server & server , std::auto_ptr<ProtocolMessage> pmessage , const std::string & ident ) :
|
||||
GNet::ServerPeer( socket , peer_address ) ,
|
||||
m_protocol( *this , thishost() , peer_address.displayString(false) ) ,
|
||||
m_pmessage( pmessage ) ,
|
||||
m_protocol( *this , m_verifier , *m_pmessage.get() , thishost() , peer_address.displayString(false) ) ,
|
||||
m_buffer( crlf() ) ,
|
||||
m_server( server )
|
||||
{
|
||||
@ -105,17 +109,19 @@ void GSmtp::ServerPeer::protocolDone()
|
||||
|
||||
// ===
|
||||
|
||||
GSmtp::Server::Server( unsigned int port , bool allow_remote , const std::string & ident ) :
|
||||
GSmtp::Server::Server( unsigned int port , bool allow_remote , const std::string & ident ,
|
||||
const std::string & downstream_server ) :
|
||||
GNet::Server( port ) ,
|
||||
m_ident( ident ) ,
|
||||
m_allow_remote( allow_remote )
|
||||
m_allow_remote( allow_remote ) ,
|
||||
m_downstream_server(downstream_server)
|
||||
{
|
||||
G_LOG( "GSmtp::Server: listening on port " << port ) ;
|
||||
//G_LOG( "GSmtp::Server: listening on port " << port ) ;
|
||||
}
|
||||
|
||||
GNet::ServerPeer * GSmtp::Server::newPeer( GNet::StreamSocket * socket , GNet::Address peer_address )
|
||||
{
|
||||
std::auto_ptr<GNet::StreamSocket> ptr(socket) ;
|
||||
std::auto_ptr<GNet::StreamSocket> socket_ptr(socket) ;
|
||||
|
||||
if( ! m_allow_remote &&
|
||||
!peer_address.sameHost(GNet::Local::canonicalAddress()) &&
|
||||
@ -128,6 +134,13 @@ GNet::ServerPeer * GSmtp::Server::newPeer( GNet::StreamSocket * socket , GNet::A
|
||||
return NULL ;
|
||||
}
|
||||
|
||||
return new ServerPeer( ptr.release() , peer_address , *this , m_ident ) ;
|
||||
const bool immediate = ! m_downstream_server.empty() ;
|
||||
|
||||
std::auto_ptr<ProtocolMessage> pmessage(
|
||||
immediate ?
|
||||
static_cast<ProtocolMessage*>(new ProtocolMessageForward(m_downstream_server)) :
|
||||
static_cast<ProtocolMessage*>(new ProtocolMessageStore) ) ;
|
||||
|
||||
return new ServerPeer( socket_ptr.release() , peer_address , *this , pmessage , m_ident ) ;
|
||||
}
|
||||
|
||||
|
@ -28,9 +28,12 @@
|
||||
#include "gsmtp.h"
|
||||
#include "gserver.h"
|
||||
#include "glinebuffer.h"
|
||||
#include "gverifier.h"
|
||||
#include "gserverprotocol.h"
|
||||
#include "gprotocolmessage.h"
|
||||
#include <string>
|
||||
#include <sstream>
|
||||
#include <memory>
|
||||
|
||||
namespace GSmtp
|
||||
{
|
||||
@ -43,10 +46,11 @@ namespace GSmtp
|
||||
// Instances are created on the heap by Server (only).
|
||||
// See also: Server
|
||||
//
|
||||
class GSmtp::ServerPeer : public GNet::ServerPeer , private GSmtp::ServerProtocol::Sender
|
||||
class GSmtp::ServerPeer : public GNet::ServerPeer , private GSmtp:: ServerProtocol::Sender
|
||||
{
|
||||
public:
|
||||
ServerPeer( GNet::StreamSocket * , GNet::Address , Server & server , const std::string & ident ) ;
|
||||
ServerPeer( GNet::StreamSocket * socket , GNet::Address address ,
|
||||
Server & server , std::auto_ptr<ProtocolMessage> pmessage , const std::string & ident ) ;
|
||||
// Constructor.
|
||||
|
||||
private:
|
||||
@ -60,10 +64,12 @@ private:
|
||||
std::string thishost() const ;
|
||||
|
||||
private:
|
||||
Server & m_server ;
|
||||
GNet::LineBuffer m_buffer ;
|
||||
static std::string crlf() ;
|
||||
ServerProtocol m_protocol ;
|
||||
Server & m_server ;
|
||||
Verifier m_verifier ; // order dependency -- first
|
||||
std::auto_ptr<ProtocolMessage> m_pmessage ; // order dependency -- second
|
||||
ServerProtocol m_protocol ; // order dependency -- third
|
||||
} ;
|
||||
|
||||
// Class: GSmtp::Server
|
||||
@ -72,8 +78,12 @@ private:
|
||||
class GSmtp::Server : private GNet::Server
|
||||
{
|
||||
public:
|
||||
Server( unsigned int port , bool allow_remote , const std::string & ident ) ;
|
||||
Server( unsigned int port , bool allow_remote , const std::string & ident ,
|
||||
const std::string & downstream_server_address ) ;
|
||||
// Constructor.
|
||||
//
|
||||
// If the 'downstream-server-address' parameter is
|
||||
// given then all messages are forwarded immediately.
|
||||
|
||||
private:
|
||||
virtual GNet::ServerPeer * newPeer( GNet::StreamSocket * , GNet::Address ) ;
|
||||
@ -83,6 +93,7 @@ private:
|
||||
private:
|
||||
std::string m_ident ;
|
||||
bool m_allow_remote ;
|
||||
std::string m_downstream_server ;
|
||||
} ;
|
||||
|
||||
#endif
|
||||
|
262
src/main/gstoredfile.cpp
Normal file
262
src/main/gstoredfile.cpp
Normal file
@ -0,0 +1,262 @@
|
||||
//
|
||||
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
//
|
||||
// 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, write to the Free Software
|
||||
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gstoredfile.cpp
|
||||
//
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gfilestore.h"
|
||||
#include "gstoredfile.h"
|
||||
#include "gmemory.h"
|
||||
#include "gfile.h"
|
||||
#include "gstr.h"
|
||||
#include "glog.h"
|
||||
#include "gassert.h"
|
||||
#include <fstream>
|
||||
|
||||
GSmtp::StoredFile::StoredFile( const G::Path & path ) :
|
||||
m_envelope_path(path)
|
||||
{
|
||||
G_DEBUG( "StoredFile: \"" << path << "\"" ) ;
|
||||
}
|
||||
|
||||
GSmtp::StoredFile::~StoredFile()
|
||||
{
|
||||
}
|
||||
|
||||
bool GSmtp::StoredFile::eightBit() const
|
||||
{
|
||||
return m_eight_bit ;
|
||||
}
|
||||
|
||||
bool GSmtp::StoredFile::readEnvelope( std::string & reason , bool check )
|
||||
{
|
||||
try
|
||||
{
|
||||
readEnvelopeCore( check ) ;
|
||||
return true ;
|
||||
}
|
||||
catch( std::exception & e )
|
||||
{
|
||||
reason = e.what() ;
|
||||
return false ;
|
||||
}
|
||||
}
|
||||
|
||||
void GSmtp::StoredFile::readEnvelopeCore( bool check )
|
||||
{
|
||||
std::ifstream stream( m_envelope_path.str().c_str() , std::ios_base::binary | std::ios_base::in ) ;
|
||||
if( ! stream.good() )
|
||||
throw OpenError() ;
|
||||
|
||||
readFormat( stream ) ;
|
||||
readFlag( stream ) ;
|
||||
readFrom( stream ) ;
|
||||
readToList( stream ) ;
|
||||
|
||||
if( check && m_to_remote.size() == 0U )
|
||||
throw NoRecipients() ;
|
||||
|
||||
if( ! stream.good() )
|
||||
throw StreamError() ;
|
||||
}
|
||||
|
||||
void GSmtp::StoredFile::readFormat( std::istream & stream )
|
||||
{
|
||||
std::string format_line = getline(stream) ;
|
||||
if( value(format_line) != FileStore::format() )
|
||||
throw InvalidFormat( value(format_line) + "!=" + FileStore::format() ) ;
|
||||
}
|
||||
|
||||
void GSmtp::StoredFile::readFlag( std::istream & stream )
|
||||
{
|
||||
std::string content_line = getline(stream) ;
|
||||
m_eight_bit = value(content_line) == "8bit" ;
|
||||
}
|
||||
|
||||
void GSmtp::StoredFile::readFrom( std::istream & stream )
|
||||
{
|
||||
m_from = value(getline(stream)) ;
|
||||
G_DEBUG( "GSmtp::StoredFile::readFrom: from \"" << m_from << "\"" ) ;
|
||||
}
|
||||
|
||||
void GSmtp::StoredFile::readToList( std::istream & stream )
|
||||
{
|
||||
std::string to_count_line = getline(stream) ;
|
||||
unsigned int to_count = G::Str::toUInt( value(to_count_line) ) ;
|
||||
|
||||
for( unsigned int i = 0U ; i < to_count ; i++ )
|
||||
{
|
||||
std::string to_line = getline(stream) ;
|
||||
bool is_local = to_line.find(FileStore::x()+"To-Local") == 0U ;
|
||||
bool is_remote = to_line.find(FileStore::x()+"To-Remote") == 0U ;
|
||||
if( ! is_local && ! is_remote )
|
||||
throw InvalidTo(to_line) ;
|
||||
|
||||
G_DEBUG( "GSmtp::StoredFile::readToList: to "
|
||||
"[" << (i+1U) << "/" << to_count << "] "
|
||||
"(" << (is_local?"local":"remote") << ") "
|
||||
<< "\"" << value(to_line) << "\"" ) ;
|
||||
|
||||
if( is_local )
|
||||
m_to_local.push_back( value(to_line) ) ;
|
||||
else
|
||||
m_to_remote.push_back( value(to_line) ) ;
|
||||
}
|
||||
}
|
||||
|
||||
void GSmtp::StoredFile::readEnd( std::istream & stream )
|
||||
{
|
||||
std::string end = getline(stream) ;
|
||||
if( end.find(FileStore::x()+"End") != 0U )
|
||||
throw NoEnd() ;
|
||||
}
|
||||
|
||||
bool GSmtp::StoredFile::openContent( std::string & reason )
|
||||
{
|
||||
try
|
||||
{
|
||||
G::Path content_path = contentPath() ;
|
||||
G_DEBUG( "GSmtp::FileStore::openContent: \"" << content_path << "\"" ) ;
|
||||
std::auto_ptr<std::istream> stream( new std::ifstream(
|
||||
content_path.str().c_str() , std::ios_base::in | std::ios_base::binary ) ) ;
|
||||
if( !stream->good() )
|
||||
{
|
||||
reason = "cannot open content file" ;
|
||||
return false ;
|
||||
}
|
||||
|
||||
G_LOG( "GSmtp::MessageStore: processing envelope \"" << m_envelope_path.basename() << "\"" ) ;
|
||||
G_LOG( "GSmtp::MessageStore: processing content \"" << content_path.basename() << "\"" ) ;
|
||||
|
||||
m_content = stream ;
|
||||
return true ;
|
||||
}
|
||||
catch( std::exception & e )
|
||||
{
|
||||
G_WARNING( "GSmtp::FileStore: exception: " << e.what() ) ;
|
||||
reason = e.what() ;
|
||||
return false ;
|
||||
}
|
||||
}
|
||||
|
||||
std::string GSmtp::StoredFile::getline( std::istream & stream ) const
|
||||
{
|
||||
return G::Str::readLineFrom( stream , crlf() ) ;
|
||||
}
|
||||
|
||||
std::string GSmtp::StoredFile::value( const std::string & s ) const
|
||||
{
|
||||
size_t pos = s.find(' ') ;
|
||||
if( pos == std::string::npos )
|
||||
throw MessageStore::FormatError() ;
|
||||
return s.substr(pos+1U) ;
|
||||
}
|
||||
|
||||
std::string GSmtp::StoredFile::crlf() const
|
||||
{
|
||||
return std::string( "\015\012" ) ;
|
||||
}
|
||||
|
||||
bool GSmtp::StoredFile::lock()
|
||||
{
|
||||
G::Path & src = m_envelope_path ;
|
||||
G::Path dst( src.str() + ".busy" ) ;
|
||||
bool ok = G::File::rename( src , dst , G::File::NoThrow() ) ;
|
||||
if( ok )
|
||||
{
|
||||
G_LOG( "GSmtp::StoredMessage: locking file \"" << src.basename() << "\"" ) ;
|
||||
m_envelope_path = dst ;
|
||||
}
|
||||
return ok ;
|
||||
}
|
||||
|
||||
void GSmtp::StoredFile::fail( const std::string & reason )
|
||||
{
|
||||
try
|
||||
{
|
||||
// write the reason into the file
|
||||
{
|
||||
std::ofstream file( m_envelope_path.str().c_str() ,
|
||||
std::ios_base::binary | std::ios_base::ate ) ;
|
||||
file << FileStore::x() << "Reason: " << reason ;
|
||||
}
|
||||
|
||||
G::Path env_temp( m_envelope_path ) ; // "foo.envelope.busy"
|
||||
env_temp.removeExtension() ; // "foo.envelope"
|
||||
G::Path bad( env_temp.str() + ".bad" ) ; // "foo.envelope.bad"
|
||||
G_LOG( "GSmtp::StoredMessage: failing file: "
|
||||
<< "\"" << m_envelope_path.basename() << "\" -> "
|
||||
<< "\"" << bad.basename() << "\"" ) ;
|
||||
|
||||
G::File::rename( m_envelope_path , bad , G::File::NoThrow() ) ;
|
||||
}
|
||||
catch(...)
|
||||
{
|
||||
}
|
||||
}
|
||||
|
||||
void GSmtp::StoredFile::destroy()
|
||||
{
|
||||
try
|
||||
{
|
||||
G_LOG( "GSmtp::StoredMessage: deleting file: \"" << m_envelope_path.basename() << "\"" ) ;
|
||||
G::File::remove( m_envelope_path , G::File::NoThrow() ) ;
|
||||
|
||||
G::Path content_path = contentPath() ;
|
||||
G_LOG( "GSmtp::StoredMessage: deleting file: \"" << content_path.basename() << "\"" ) ;
|
||||
m_content <<= 0 ; // close it first
|
||||
G::File::remove( content_path , G::File::NoThrow() ) ;
|
||||
}
|
||||
catch(...)
|
||||
{
|
||||
}
|
||||
}
|
||||
|
||||
const std::string & GSmtp::StoredFile::from() const
|
||||
{
|
||||
return m_from ;
|
||||
}
|
||||
|
||||
const G::Strings & GSmtp::StoredFile::to() const
|
||||
{
|
||||
return m_to_remote ;
|
||||
}
|
||||
|
||||
std::auto_ptr<std::istream> GSmtp::StoredFile::extractContentStream()
|
||||
{
|
||||
G_ASSERT( m_content.get() != NULL ) ;
|
||||
return m_content ;
|
||||
}
|
||||
|
||||
G::Path GSmtp::StoredFile::contentPath() const
|
||||
{
|
||||
G::Path path( m_envelope_path ) ; // "foo.envelope.busy"
|
||||
path.removeExtension() ; // "foo.envelope"
|
||||
path.setExtension( "content" ) ; // "foo.content"
|
||||
return path ;
|
||||
}
|
||||
|
||||
size_t GSmtp::StoredFile::remoteRecipientCount() const
|
||||
{
|
||||
return m_to_remote.size() ;
|
||||
}
|
||||
|
119
src/main/gstoredfile.h
Normal file
119
src/main/gstoredfile.h
Normal file
@ -0,0 +1,119 @@
|
||||
//
|
||||
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
//
|
||||
// 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, write to the Free Software
|
||||
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gstoredfile.h
|
||||
//
|
||||
|
||||
#ifndef G_SMTP_STORED_FILE_H
|
||||
#define G_SMTP_STORED_FILE_H
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gmessagestore.h"
|
||||
#include "gstoredmessage.h"
|
||||
#include "gexception.h"
|
||||
#include "gpath.h"
|
||||
#include "gstrings.h"
|
||||
#include <iostream>
|
||||
#include <memory>
|
||||
|
||||
namespace GSmtp
|
||||
{
|
||||
class StoredFile ;
|
||||
} ;
|
||||
|
||||
// Class: GSmtp::StoredFile
|
||||
// Description: A concete derived class implementing the
|
||||
// StoredMessage interface.
|
||||
//
|
||||
class GSmtp::StoredFile : public GSmtp:: StoredMessage
|
||||
{
|
||||
public:
|
||||
G_EXCEPTION( InvalidFormat , "invalid format field in envelope" ) ;
|
||||
G_EXCEPTION( NoEnd , "invalid envelope file: no end marker" ) ;
|
||||
G_EXCEPTION( InvalidTo , "invalid 'to' line in envelope file" ) ;
|
||||
G_EXCEPTION( NoRecipients , "no remote recipients" ) ;
|
||||
G_EXCEPTION( OpenError , "cannot open the envelope" ) ;
|
||||
G_EXCEPTION( StreamError , "envelope reading/parsing error" ) ;
|
||||
|
||||
explicit StoredFile( const G::Path & envelope_path ) ;
|
||||
// Constructor.
|
||||
|
||||
virtual ~StoredFile() ;
|
||||
// Destructor.
|
||||
|
||||
bool lock() ;
|
||||
// Locks the file by renaming the envelope file.
|
||||
// Used by FileStore and FileIterator.
|
||||
|
||||
bool readEnvelope( std::string & reason , bool check_for_no_remote_recipients ) ;
|
||||
// Reads the envelope. Returns false on error.
|
||||
// Used by FileStore and FileIterator.
|
||||
|
||||
bool openContent( std::string & reason ) ;
|
||||
// Opens the content file. Returns false on error.
|
||||
// Used by FileStore and FileIterator.
|
||||
|
||||
virtual bool eightBit() const ;
|
||||
// From StoredMessage.
|
||||
|
||||
virtual const std::string & from() const ;
|
||||
// From StoredMessage.
|
||||
|
||||
virtual const G::Strings & to() const ;
|
||||
// From StoredMessage.
|
||||
|
||||
virtual void destroy() ;
|
||||
// From StoredMessage.
|
||||
|
||||
virtual void fail( const std::string & reason ) ;
|
||||
// From StoredMessage.
|
||||
|
||||
virtual std::auto_ptr<std::istream> extractContentStream() ;
|
||||
// From StoredMessage.
|
||||
|
||||
virtual size_t remoteRecipientCount() const ;
|
||||
// From StoredMessage.
|
||||
|
||||
private:
|
||||
StoredFile( const StoredFile & ) ;
|
||||
void operator=( const StoredFile & ) ;
|
||||
std::string crlf() const ;
|
||||
std::string getline( std::istream & stream ) const ;
|
||||
std::string value( const std::string & s ) const ;
|
||||
G::Path contentPath() const ;
|
||||
void readFormat( std::istream & stream ) ;
|
||||
void readFlag( std::istream & stream ) ;
|
||||
void readFrom( std::istream & stream ) ;
|
||||
void readToList( std::istream & stream ) ;
|
||||
void readEnd( std::istream & stream ) ;
|
||||
void readEnvelopeCore( bool ) ;
|
||||
|
||||
private:
|
||||
G::Strings m_to_local ;
|
||||
G::Strings m_to_remote ;
|
||||
std::string m_from ;
|
||||
G::Path m_envelope_path ;
|
||||
std::auto_ptr<std::istream> m_content ;
|
||||
bool m_eight_bit ;
|
||||
} ;
|
||||
|
||||
#endif
|
||||
|
@ -18,64 +18,15 @@
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gpid_unix.cpp
|
||||
// gstoredmessage.cpp
|
||||
//
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gpid.h"
|
||||
#include <unistd.h>
|
||||
#include <sys/types.h>
|
||||
#include <sstream>
|
||||
#include "gsmtp.h"
|
||||
#include "gstoredmessage.h"
|
||||
|
||||
namespace G
|
||||
GSmtp::StoredMessage::~StoredMessage()
|
||||
{
|
||||
class PidImp ;
|
||||
} ;
|
||||
|
||||
// Class: G::PidImp
|
||||
// Description: A pimple implementation class for GPid.
|
||||
//
|
||||
class G::PidImp
|
||||
{
|
||||
public:
|
||||
pid_t m_pid ;
|
||||
} ;
|
||||
|
||||
// ===
|
||||
|
||||
G::Pid::Pid() : m_imp(NULL)
|
||||
{
|
||||
m_imp = new PidImp ;
|
||||
m_imp->m_pid = ::getpid() ;
|
||||
}
|
||||
|
||||
G::Pid::~Pid()
|
||||
{
|
||||
delete m_imp ;
|
||||
}
|
||||
|
||||
G::Pid::Pid( const Pid & other ) :
|
||||
m_imp(NULL)
|
||||
{
|
||||
m_imp = new PidImp ;
|
||||
m_imp->m_pid = other.m_imp->m_pid ;
|
||||
}
|
||||
|
||||
G::Pid & G::Pid::operator=( const Pid & rhs )
|
||||
{
|
||||
m_imp->m_pid = rhs.m_imp->m_pid ;
|
||||
return *this ;
|
||||
}
|
||||
|
||||
std::string G::Pid::str() const
|
||||
{
|
||||
std::stringstream ss ;
|
||||
ss << m_imp->m_pid ;
|
||||
return ss.str() ;
|
||||
}
|
||||
|
||||
bool G::Pid::operator==( const Pid & rhs ) const
|
||||
{
|
||||
return m_imp->m_pid == rhs.m_imp->m_pid ;
|
||||
// empty
|
||||
}
|
||||
|
77
src/main/gstoredmessage.h
Normal file
77
src/main/gstoredmessage.h
Normal file
@ -0,0 +1,77 @@
|
||||
//
|
||||
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
//
|
||||
// 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, write to the Free Software
|
||||
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gstoredmessage.h
|
||||
//
|
||||
|
||||
#ifndef G_SMTP_STORED_MESSAGE_H
|
||||
#define G_SMTP_STORED_MESSAGE_H
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gstrings.h"
|
||||
#include "gpath.h"
|
||||
|
||||
namespace GSmtp
|
||||
{
|
||||
class StoredMessage ;
|
||||
} ;
|
||||
|
||||
// Class: GSmtp::StoredMessage
|
||||
// Description: An abstract class for messages which have
|
||||
// come from the store.
|
||||
// See also: MessageStore, MessageStore::get()
|
||||
//
|
||||
class GSmtp::StoredMessage
|
||||
{
|
||||
public:
|
||||
virtual const std::string & from() const = 0 ;
|
||||
// Returns the envelope 'from' field.
|
||||
|
||||
virtual const G::Strings & to() const = 0 ;
|
||||
// Returns the envelope 'to' fields.
|
||||
|
||||
virtual std::auto_ptr<std::istream> extractContentStream() = 0 ;
|
||||
// Extracts the content stream.
|
||||
// Can only be called once.
|
||||
|
||||
virtual void destroy() = 0 ;
|
||||
// Deletes the message within the store.
|
||||
|
||||
virtual void fail( const std::string & reason ) = 0 ;
|
||||
// Marks the message as failed within the store.
|
||||
|
||||
virtual bool eightBit() const = 0 ;
|
||||
// Returns true if the message content (header+body)
|
||||
// contains a character with the most significant
|
||||
// bit set.
|
||||
|
||||
virtual size_t remoteRecipientCount() const = 0 ;
|
||||
// Returns the number of non-local recipients.
|
||||
|
||||
virtual ~StoredMessage() ;
|
||||
// Destructor.
|
||||
|
||||
private:
|
||||
void operator=( const StoredMessage & ) ; // not implemented
|
||||
} ;
|
||||
|
||||
#endif
|
||||
|
66
src/main/gverifier.cpp
Normal file
66
src/main/gverifier.cpp
Normal file
@ -0,0 +1,66 @@
|
||||
//
|
||||
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
//
|
||||
// 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, write to the Free Software
|
||||
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gverifier.cpp
|
||||
//
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include "gverifier.h"
|
||||
#include "gstr.h"
|
||||
#include "gassert.h"
|
||||
#include "glog.h"
|
||||
|
||||
GSmtp::Verifier::Status GSmtp::Verifier::verify( const std::string & user ) const
|
||||
{
|
||||
G_DEBUG( "GSmtp::ProtocolMessage::verify: \"" << user << "\"" ) ;
|
||||
Status rc( isLocal(user) , std::string() ) ;
|
||||
if( isLocal(user) && isValid(user) )
|
||||
rc.second = fullName(user) ;
|
||||
return rc ;
|
||||
}
|
||||
|
||||
//static
|
||||
bool GSmtp::Verifier::isLocal( const std::string & user )
|
||||
{
|
||||
return user.find('@') == std::string::npos ;
|
||||
}
|
||||
|
||||
//static
|
||||
bool GSmtp::Verifier::isValid( const std::string & user )
|
||||
{
|
||||
// only recognise one local mailbox
|
||||
return isPostmaster(user) ;
|
||||
}
|
||||
|
||||
//static
|
||||
bool GSmtp::Verifier::isPostmaster( std::string user )
|
||||
{
|
||||
G::Str::toUpper( user ) ;
|
||||
G::Str::trim( user , " \t" ) ;
|
||||
return user == "POSTMASTER" ;
|
||||
}
|
||||
|
||||
//static
|
||||
std::string GSmtp::Verifier::fullName( const std::string & user )
|
||||
{
|
||||
return "Local postmaster <postmaster@localhost>" ;
|
||||
}
|
||||
|
64
src/main/gverifier.h
Normal file
64
src/main/gverifier.h
Normal file
@ -0,0 +1,64 @@
|
||||
//
|
||||
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
|
||||
//
|
||||
// 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, write to the Free Software
|
||||
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
//
|
||||
// ===
|
||||
//
|
||||
// gverifier.h
|
||||
//
|
||||
|
||||
#ifndef G_SMTP_VERIFIER_H
|
||||
#define G_SMTP_VERIFIER_H
|
||||
|
||||
#include "gdef.h"
|
||||
#include "gsmtp.h"
|
||||
#include <string>
|
||||
|
||||
namespace GSmtp
|
||||
{
|
||||
class Verifier ;
|
||||
} ;
|
||||
|
||||
// Class: GSmtp::Verifier
|
||||
// Description: A class which verifies recipient addresses.
|
||||
// This functionality is used in the VRFY and RCPT commands
|
||||
// in the SMTP server-side protocol.
|
||||
// See also: ServerProtocol
|
||||
//
|
||||
class GSmtp::Verifier
|
||||
{
|
||||
public:
|
||||
typedef std::pair<bool,std::string> Status ;
|
||||
|
||||
std::pair<bool,std::string> verify( const std::string & recipient_address ) const ;
|
||||
// Checks a recipient address returning
|
||||
// <is-local>|<local-full-name>.
|
||||
//
|
||||
// If syntactically local then 'first' is
|
||||
// returned true. If local and valid then
|
||||
// 'second' is set to the full description.
|
||||
// If syntactically remote, then 'first'
|
||||
// is returned false and 'second' is empty.
|
||||
|
||||
private:
|
||||
static bool isLocal( const std::string & user ) ;
|
||||
static bool isValid( const std::string & user ) ;
|
||||
static bool isPostmaster( std::string user ) ;
|
||||
static std::string fullName( const std::string & user ) ;
|
||||
} ;
|
||||
|
||||
#endif
|
BIN
src/main/icon-32.ico
Normal file
BIN
src/main/icon-32.ico
Normal file
Binary file not shown.
After Width: | Height: | Size: 766 B |
@ -30,9 +30,12 @@
|
||||
#include "gdaemon.h"
|
||||
#include "gstr.h"
|
||||
#include "gpath.h"
|
||||
#include "gmessagestore.h"
|
||||
#include "gfilestore.h"
|
||||
#include "gnewfile.h"
|
||||
#include "gadminserver.h"
|
||||
#include "gexception.h"
|
||||
#include "gprocess.h"
|
||||
#include "gmemory.h"
|
||||
#include "ggetopt.h"
|
||||
#include "gdebug.h"
|
||||
#include <iostream>
|
||||
@ -62,6 +65,7 @@ namespace
|
||||
unsigned int optPort() const ;
|
||||
unsigned int optAdminPort() const ;
|
||||
bool optCloseStderr() const ;
|
||||
bool optImmediate() const ;
|
||||
bool optLog() const ;
|
||||
bool optSyslog() const ;
|
||||
bool optDaemon() const ;
|
||||
@ -102,8 +106,8 @@ void Main::warranty() const
|
||||
{
|
||||
std::cout
|
||||
<< "This software is provided without warranty of any kind." << std::endl
|
||||
<< "You may redistribure copies of this program under the terms of the GNU "
|
||||
<< "General Public License." << std::endl
|
||||
<< "You may redistribure copies of this program under " << std::endl
|
||||
<< "the terms of the GNU General Public License." << std::endl
|
||||
<< "For more information refer to the file named COPYING." << std::endl ;
|
||||
}
|
||||
|
||||
@ -116,7 +120,7 @@ void Main::version() const
|
||||
|
||||
std::string Main::versionNumber() const
|
||||
{
|
||||
return "0.9.2" ;
|
||||
return "0.9.3" ;
|
||||
}
|
||||
|
||||
std::string Main::smtpIdent() const
|
||||
@ -130,7 +134,10 @@ unsigned int Main::ttyColumns() const
|
||||
try
|
||||
{
|
||||
const char * p = std::getenv( "COLUMNS" ) ;
|
||||
return p ? G::Str::toUInt(p) : default_ ;
|
||||
if( p == NULL )
|
||||
return default_ ;
|
||||
else
|
||||
return G::Str::toUInt(p) ;
|
||||
}
|
||||
catch( std::exception & )
|
||||
{
|
||||
@ -158,9 +165,8 @@ void Main::help( const std::string & exe ) const
|
||||
<< std::endl ;
|
||||
|
||||
std::cout
|
||||
<< "To start a 'store & forward' daemon..." << std::endl
|
||||
<< " " << exe << " --as-server --admin 10025 --forward-to mail.myisp.co.uk:smtp" << std::endl
|
||||
<< " (and then \"" << exe << "poke 10025\" to trigger forwarding)" << std::endl
|
||||
<< "To run as a proxy (on port 10025) to a local server (on port 25)..." << std::endl
|
||||
<< " " << exe << " --port 10025 --as-proxy localhost:25" << std::endl
|
||||
<< std::endl ;
|
||||
}
|
||||
|
||||
@ -186,6 +192,7 @@ std::string Main::switchSpec() const
|
||||
<< "q!as-client!equivalent to \"--no-syslog --no-daemon --log --dont-serve --forward --forward-to\"!"
|
||||
<< "1!host:port|"
|
||||
<< "d!as-server!equivalent to \"--close-stderr --log\"!0!|"
|
||||
<< "m!immediate!forwards each message as soon as it is received (requires --forward-to)!0!|"
|
||||
<< "n!no-syslog!disables syslog output!0!|"
|
||||
<< "t!no-daemon!does not detach from the terminal!0!|"
|
||||
<< "x!dont-serve!stops the process acting as a server (usually used with --forward)!0!|"
|
||||
@ -195,6 +202,8 @@ std::string Main::switchSpec() const
|
||||
<< "i!pid-file!records the daemon process-id in the given file!1!pid-file|"
|
||||
<< "p!port!specifies the smtp listening port number!1!port|"
|
||||
<< "a!admin!enables the administration interface and specifies its listening port number!1!admin-port|"
|
||||
<< "y!as-proxy!equivalent to \"--close-stderr --log --immediate --forward-to\"!1!host:port|"
|
||||
<< "z!filter!defines a mail pre-processor (disallowed if running as root)!1!program|"
|
||||
<< "V!version!displays version information and exits!0!"
|
||||
;
|
||||
return ss.str() ;
|
||||
@ -244,7 +253,11 @@ void Main::run()
|
||||
|
||||
bool Main::optLog() const
|
||||
{
|
||||
return opt().contains("log") || opt().contains("as-client") || opt().contains("as-server") ;
|
||||
return
|
||||
opt().contains("log") ||
|
||||
opt().contains("as-client") ||
|
||||
opt().contains("as-proxy") ||
|
||||
opt().contains("as-server") ;
|
||||
}
|
||||
|
||||
bool Main::optSyslog() const
|
||||
@ -266,7 +279,17 @@ unsigned int Main::optAdminPort() const
|
||||
|
||||
bool Main::optCloseStderr() const
|
||||
{
|
||||
return opt().contains("close-stderr") || opt().contains("as-server") ;
|
||||
return
|
||||
opt().contains("close-stderr") ||
|
||||
opt().contains("as-proxy") ||
|
||||
opt().contains("as-server") ;
|
||||
}
|
||||
|
||||
bool Main::optImmediate() const
|
||||
{
|
||||
return
|
||||
opt().contains("immediate") ||
|
||||
opt().contains("as-proxy") ;
|
||||
}
|
||||
|
||||
bool Main::optDaemon() const
|
||||
@ -283,7 +306,11 @@ G::Path Main::optSpoolDir() const
|
||||
|
||||
std::string Main::optServerAddress() const
|
||||
{
|
||||
const char * key = opt().contains("forward-to") ? "forward-to" : "as-client" ;
|
||||
const char * key = "forward-to" ;
|
||||
if( opt().contains("as-client") )
|
||||
key = "as-client" ;
|
||||
else if( opt().contains("as-proxy") )
|
||||
key = "as-proxy" ;
|
||||
return opt().contains(key) ? opt().value(key) : std::string() ;
|
||||
}
|
||||
|
||||
@ -301,10 +328,12 @@ std::string Main::checkOptions() const
|
||||
"be an absolute path (starting with /)" ;
|
||||
}
|
||||
|
||||
if( !opt().contains("forward-to") &&
|
||||
(opt().contains("admin") || opt().contains("forward")) )
|
||||
if( !opt().contains("forward-to") && (
|
||||
opt().contains("forward") ||
|
||||
opt().contains("immediate") ||
|
||||
opt().contains("admin") ) )
|
||||
{
|
||||
return "usage error: the --admin and --forward "
|
||||
return "usage error: the --forward, --immediate and --admin "
|
||||
"switches require --forward-to" ;
|
||||
}
|
||||
|
||||
@ -332,14 +361,14 @@ void Main::closeFiles()
|
||||
if( optDaemon() )
|
||||
{
|
||||
const bool keep_stderr = true ;
|
||||
G::Daemon::closeFiles( keep_stderr ) ;
|
||||
G::Process::closeFiles( keep_stderr ) ;
|
||||
}
|
||||
}
|
||||
|
||||
void Main::closeMoreFiles()
|
||||
{
|
||||
if( optDaemon() && optCloseStderr() )
|
||||
G::Daemon::closeStderr() ;
|
||||
G::Process::closeStderr() ;
|
||||
}
|
||||
|
||||
bool Main::optDoServing() const
|
||||
@ -363,17 +392,19 @@ void Main::runCore()
|
||||
if( !error.empty() )
|
||||
throw G::Exception( error ) ;
|
||||
|
||||
G::Daemon::setUmask() ;
|
||||
G::Daemon::PidFile pid_file ;
|
||||
G::Process::setUmask() ;
|
||||
if( optDaemon() )
|
||||
{
|
||||
closeFiles() ; // before opening any sockets or message-store streams
|
||||
if( opt().contains("pid-file") )
|
||||
G::Daemon::detach( G::Path(opt().value("pid-file")) ) ;
|
||||
else
|
||||
G::Daemon::detach() ;
|
||||
pid_file = G::Daemon::PidFile( G::Path(opt().value("pid-file")) ) ;
|
||||
G::Daemon::detach( pid_file ) ;
|
||||
}
|
||||
|
||||
GSmtp::MessageStore store( optSpoolDir() ) ;
|
||||
GSmtp::FileStore store( optSpoolDir() ) ;
|
||||
if( opt().contains("filter") )
|
||||
GSmtp::NewFile::setPreprocessor( G::Path(opt().value("filter")) ) ;
|
||||
|
||||
std::auto_ptr<GNet::EventSources> event_loop( GNet::EventSources::create() ) ;
|
||||
if( ! event_loop->init() )
|
||||
@ -389,21 +420,20 @@ void Main::runCore()
|
||||
|
||||
if( optDoServing() )
|
||||
{
|
||||
GSmtp::Server server( optPort() , optAllowRemoteClients() , smtpIdent() ) ;
|
||||
GSmtp::Server server( optPort() , optAllowRemoteClients() , smtpIdent() ,
|
||||
optImmediate() ? optServerAddress() : std::string() ) ;
|
||||
|
||||
std::auto_ptr<GSmtp::AdminServer> admin_server ;
|
||||
if( optDoAdmin() )
|
||||
{
|
||||
GSmtp::AdminServer admin_server( optAdminPort() ,
|
||||
admin_server <<= new GSmtp::AdminServer( optAdminPort() ,
|
||||
optAllowRemoteClients() , optServerAddress() ) ;
|
||||
}
|
||||
|
||||
pid_file.commit() ;
|
||||
closeMoreFiles() ;
|
||||
event_loop->run() ;
|
||||
}
|
||||
else
|
||||
{
|
||||
closeMoreFiles() ;
|
||||
event_loop->run() ;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
int main( int argc , char * argv [] )
|
||||
|
@ -50,9 +50,13 @@ int main( int argc , char * argv [] )
|
||||
struct sockaddr_in address ;
|
||||
int fd , rc ;
|
||||
|
||||
/* parse the command line -- port number */
|
||||
if( argc > 1 )
|
||||
{
|
||||
port = atoi(argv[1]) ;
|
||||
}
|
||||
|
||||
/* parse the command line -- send string */
|
||||
if( argc > 2 )
|
||||
{
|
||||
buffer[0] = '\0' ;
|
||||
@ -60,22 +64,38 @@ int main( int argc , char * argv [] )
|
||||
}
|
||||
strcat( buffer , "\015\012" ) ;
|
||||
|
||||
/* open the socket */
|
||||
fd = socket( AF_INET , SOCK_STREAM , 0 ) ;
|
||||
if( fd < 0 ) return EXIT_FAILURE ;
|
||||
if( fd < 0 )
|
||||
return EXIT_FAILURE ;
|
||||
|
||||
/* prepare the address */
|
||||
memset( &address , 0 , sizeof(address) ) ;
|
||||
address.sin_family = AF_INET ;
|
||||
address.sin_port = htons( port ) ;
|
||||
address.sin_addr.s_addr = inet_addr( host ) ;
|
||||
|
||||
/* connect */
|
||||
rc = connect( fd , (const struct sockaddr*)&address , sizeof(address) ) ;
|
||||
if( rc < 0 ) return EXIT_FAILURE ;
|
||||
if( rc < 0 )
|
||||
return EXIT_FAILURE ;
|
||||
|
||||
/* send the string */
|
||||
rc = write( fd , buffer , strlen(buffer) ) ;
|
||||
if( rc != strlen(buffer) ) return EXIT_FAILURE ;
|
||||
if( rc != strlen(buffer) )
|
||||
return EXIT_FAILURE ;
|
||||
|
||||
/* read the reply */
|
||||
rc = read( fd , buffer , sizeof(buffer)-1U) ;
|
||||
if( rc <= 0 ) return EXIT_FAILURE ;
|
||||
if( rc <= 0 )
|
||||
return EXIT_FAILURE ;
|
||||
|
||||
/* print the reply */
|
||||
write( STDOUT_FILENO , buffer , rc ) ;
|
||||
buffer[0U] = '\n' ;
|
||||
buffer[1U] = '\0' ;
|
||||
write( STDOUT_FILENO , buffer , strlen(buffer) ) ;
|
||||
|
||||
return EXIT_SUCCESS ;
|
||||
}
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user