This commit is contained in:
Graeme Walker 2002-08-16 12:00:00 +00:00
parent 38eb15ca53
commit 0a46d723c5
194 changed files with 5471 additions and 2794 deletions

View File

@ -1,6 +1,19 @@
E-MailRelay Change Log
======================
0.9.9 -> 1.0.0
--------------
* Briefer "--help" output; works with "--verbose".
* Option to listen on a specific network interface ("--interface").
* Option for an external address verifier program ("--verifier").
* Some Linux Standard Base stuff added to the "init.d" script.
* Pid files world-readable and deleted on abnormal termination.
* Compiles with gcc 3.0 and intel 6.0.
* Autoconf tweak for MacOS X.
* Corrected the "Received:" typo [bug-id 572236].
* EHLO response parsing is now case-insensitive [bug-id 561522].
* Fewer missing-secrets warnings [bug-id 564987].
0.9.8 -> 0.9.9
--------------
* More flexible logging options ("--verbose" and "--debug" work better).

19
INSTALL
View File

@ -1,16 +1,20 @@
Introduction
============
What follows are generic installation instructions for doing a standard GNU
"./configure; make; make install" installation from source under Linux, FreeBSD
etc. The Windows installation instructions are in a separate document.
Note that a non-standard "configure" switch is available, "--enable-fhs",
which overrides all other directory modifiers, forcing compliance with the File
Hierarchy Standard. There are also a set of variables which can be defined on
the "configure" command line for controlling all the installation directories
in more detail, augmenting the standard command-line switches like "--sbindir"
and "--libexecdir". For more information refer to the E-MailRelay reference
document.
A non-standard "configure" switch is available, "--enable-fhs", which overrides
all other directory modifiers, forcing compliance with the File Hierarchy Standard.
This switch is used in building the RPMs, and may become the default in future
releases.
There are also a set of variables which can be defined on the "configure" command
line for controlling all the installation directories in more detail, augmenting
the standard command-line switches like "--sbindir" and "--libexecdir". For more
information refer to the E-MailRelay reference document under
"Files and directories".
The E-MailRelay user guide describes what needs to be done after the "make
install" in order to get the emailrelay daemon to start up at boot-time,
@ -18,6 +22,7 @@ automatically forward e-mail and bounce failed mail.
Basic Installation
==================
The `configure' shell script attempts to guess correct values for
various system-dependent variables used during compilation. It uses
those values to create a `Makefile' in each directory of the package.

View File

@ -21,4 +21,6 @@ SUBDIRS = src bin lib doc
e_doc_DATA = NEWS README changelog.gz
CLEANFILES = changelog.gz
changelog.gz: ChangeLog
if test -n "$(GZIP)" ; then $(GZIP) -c $< > changelog.tmp && mv changelog.tmp changelog.gz ; fi
if test -n "$(GZIP)" ; then $(GZIP) -c $(top_srcdir)/ChangeLog > changelog.tmp && mv changelog.tmp changelog.gz ; fi
uninstall-local:
-rmdir $(DESTDIR)$(e_docdir) 2>/dev/null

View File

@ -79,6 +79,7 @@ POST_UNINSTALL = :
AR = @AR@
AWK = @AWK@
CC = @CC@
COMPILER_VERSION = @COMPILER_VERSION@
CXX = @CXX@
GZIP = @GZIP@
HAVE_DOXYGEN = @HAVE_DOXYGEN@
@ -346,7 +347,7 @@ install-data: install-data-recursive
install-am: all-am
@$(MAKE) $(AM_MAKEFLAGS) install-exec-am install-data-am
install: install-recursive
uninstall-am: uninstall-e_docDATA
uninstall-am: uninstall-e_docDATA uninstall-local
uninstall: uninstall-recursive
all-am: Makefile $(DATA) config.h
all-redirect: all-recursive-am
@ -398,12 +399,15 @@ maintainer-clean-recursive tags tags-recursive mostlyclean-tags \
distclean-tags clean-tags maintainer-clean-tags distdir info-am info \
dvi-am dvi check check-am installcheck-am installcheck all-recursive-am \
install-exec-am install-exec install-data-am install-data install-am \
install uninstall-am uninstall all-redirect all-am all installdirs-am \
installdirs mostlyclean-generic distclean-generic clean-generic \
maintainer-clean-generic clean mostlyclean distclean maintainer-clean
install uninstall-local uninstall-am uninstall all-redirect all-am all \
installdirs-am installdirs mostlyclean-generic distclean-generic \
clean-generic maintainer-clean-generic clean mostlyclean distclean \
maintainer-clean
changelog.gz: ChangeLog
if test -n "$(GZIP)" ; then $(GZIP) -c $< > changelog.tmp && mv changelog.tmp changelog.gz ; fi
if test -n "$(GZIP)" ; then $(GZIP) -c $(top_srcdir)/ChangeLog > changelog.tmp && mv changelog.tmp changelog.gz ; fi
uninstall-local:
-rmdir $(DESTDIR)$(e_docdir) 2>/dev/null
# Tell versions [3.59,3.63) of GNU make to not export all variables.
# Otherwise a system limit (for SysV at least) may be exceeded.

8
NEWS
View File

@ -1,7 +1 @@
To do...
* optionally enforce authentication by clients
* OS-X port
* better windows gui (esp. client side)
* sample pre-processor script for spamassassin
no news

9
README
View File

@ -12,8 +12,8 @@ than to a local postmaster. Because of this functional simplicity it is
extremely easy to configure, typically only requiring the address of the
next-hop 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.
C++ source code is available for Linux, FreeBSD (etc) and Windows.
Distribution is under the GNU General Public License.
Quick start
-----------
@ -78,11 +78,12 @@ and ported to Windows 98 using:
The code has also been built successfully on:
* Windows NT 4.0
* MacOS X
* FreeBSD on Intel hardware
* Solaris 8, using gcc, on Sparc hardware
* 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
Feedback

View File

@ -18,6 +18,62 @@ dnl Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
dnl
dnl ===
dnl derived from lars brinkhoff...
AC_DEFUN(ACLOCAL_TYPE_SOCKLEN_T,
[AC_CACHE_CHECK([for socklen_t], aclocal_cv_type_socklen_t,
[
AC_TRY_COMPILE(
[#include <sys/types.h>
#include <sys/socket.h>],
[socklen_t len = 42; return len;],
aclocal_cv_type_socklen_t=yes,
aclocal_cv_type_socklen_t=no)
])
if test $aclocal_cv_type_socklen_t = yes; then
AC_DEFINE(HAVE_SOCKLEN_T, 1,[Define if socklen_t type definition in sys/socket.h])
else
AC_DEFINE(HAVE_SOCKLEN_T, 0,[Define if socklen_t type definition in sys/socket.h])
fi
])
AC_DEFUN([ACLOCAL_CHECK_GMTIME_R],
[AC_CACHE_CHECK([for gmtime_r], aclocal_cv_gmtime_r,
[
AC_TRY_COMPILE(
[#include <time.h>],
[gmtime_r((time_t*)0,(struct tm*)0) ;],
aclocal_cv_gmtime_r=yes ,
aclocal_cv_gmtime_r=no )
])
if test $aclocal_cv_gmtime_r = yes; then
AC_DEFINE(HAVE_GMTIME_R,1,[Define if gmtime_r in time.h])
else
AC_DEFINE(HAVE_GMTIME_R,0,[Define if gmtime_r in time.h])
fi
])
AC_DEFUN([ACLOCAL_CHECK_LOCALTIME_R],
[AC_CACHE_CHECK([for localtime_r], aclocal_cv_localtime_r,
[
AC_TRY_COMPILE(
[#include <time.h>],
[localtime_r((time_t*)0,(struct tm*)0) ;],
aclocal_cv_localtime_r=yes ,
aclocal_cv_localtime_r=no )
])
if test $aclocal_cv_localtime_r = yes; then
AC_DEFINE(HAVE_LOCALTIME_R,1,[Define if localtime_r in time.h])
else
AC_DEFINE(HAVE_LOCALTIME_R,0,[Define if localtime_r in time.h])
fi
])
AC_DEFUN([ACLOCAL_COMPILER_VERSION],
[
COMPILER_VERSION=`$CXX --version 2>/dev/null | sed 's/\./ /;s/\..*//;s/ /\./;s/ .*//;s/^/gcc/'`
AC_SUBST(COMPILER_VERSION)
])
AC_DEFUN([ENABLE_FHS],
[
if test "$enable_fhs" = "yes"

56
aclocal.m4 vendored
View File

@ -30,6 +30,62 @@ dnl Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
dnl
dnl ===
dnl derived from lars brinkhoff...
AC_DEFUN(ACLOCAL_TYPE_SOCKLEN_T,
[AC_CACHE_CHECK([for socklen_t], aclocal_cv_type_socklen_t,
[
AC_TRY_COMPILE(
[#include <sys/types.h>
#include <sys/socket.h>],
[socklen_t len = 42; return len;],
aclocal_cv_type_socklen_t=yes,
aclocal_cv_type_socklen_t=no)
])
if test $aclocal_cv_type_socklen_t = yes; then
AC_DEFINE(HAVE_SOCKLEN_T, 1,[Define if socklen_t type definition in sys/socket.h])
else
AC_DEFINE(HAVE_SOCKLEN_T, 0,[Define if socklen_t type definition in sys/socket.h])
fi
])
AC_DEFUN([ACLOCAL_CHECK_GMTIME_R],
[AC_CACHE_CHECK([for gmtime_r], aclocal_cv_gmtime_r,
[
AC_TRY_COMPILE(
[#include <time.h>],
[gmtime_r((time_t*)0,(struct tm*)0) ;],
aclocal_cv_gmtime_r=yes ,
aclocal_cv_gmtime_r=no )
])
if test $aclocal_cv_gmtime_r = yes; then
AC_DEFINE(HAVE_GMTIME_R,1,[Define if gmtime_r in time.h])
else
AC_DEFINE(HAVE_GMTIME_R,0,[Define if gmtime_r in time.h])
fi
])
AC_DEFUN([ACLOCAL_CHECK_LOCALTIME_R],
[AC_CACHE_CHECK([for localtime_r], aclocal_cv_localtime_r,
[
AC_TRY_COMPILE(
[#include <time.h>],
[localtime_r((time_t*)0,(struct tm*)0) ;],
aclocal_cv_localtime_r=yes ,
aclocal_cv_localtime_r=no )
])
if test $aclocal_cv_localtime_r = yes; then
AC_DEFINE(HAVE_LOCALTIME_R,1,[Define if localtime_r in time.h])
else
AC_DEFINE(HAVE_LOCALTIME_R,0,[Define if localtime_r in time.h])
fi
])
AC_DEFUN([ACLOCAL_COMPILER_VERSION],
[
COMPILER_VERSION=`$CXX --version 2>/dev/null | sed 's/\./ /;s/\..*//;s/ /\./;s/ .*//;s/^/gcc/'`
AC_SUBST(COMPILER_VERSION)
])
AC_DEFUN([ENABLE_FHS],
[
if test "$enable_fhs" = "yes"

View File

@ -37,9 +37,12 @@ emailrelay: emailrelay.sh
cp emailrelay.sh emailrelay
chmod ugo+x emailrelay
install-data-local:
install-data-local: install-e_examplesDATA install-e_initSCRIPTS
chmod ugo+x $(DESTDIR)$(e_examplesdir)/emailrelay-notify.sh
chmod ugo+x $(DESTDIR)$(e_examplesdir)/emailrelay-resubmit.sh
chmod ugo+x $(DESTDIR)$(e_examplesdir)/emailrelay-deliver.sh
chmod ugo+x $(DESTDIR)$(e_examplesdir)/emailrelay-process.sh
uninstall-local:
-rmdir $(DESTDIR)$(e_examplesdir) 2>/dev/null

View File

@ -79,6 +79,7 @@ POST_UNINSTALL = :
AR = @AR@
AWK = @AWK@
CC = @CC@
COMPILER_VERSION = @COMPILER_VERSION@
CXX = @CXX@
GZIP = @GZIP@
HAVE_DOXYGEN = @HAVE_DOXYGEN@
@ -232,7 +233,8 @@ install-data: install-data-am
install-am: all-am
@$(MAKE) $(AM_MAKEFLAGS) install-exec-am install-data-am
install: install-am
uninstall-am: uninstall-e_initSCRIPTS uninstall-e_examplesDATA
uninstall-am: uninstall-e_initSCRIPTS uninstall-e_examplesDATA \
uninstall-local
uninstall: uninstall-am
all-am: Makefile $(SCRIPTS) $(DATA)
all-redirect: all-am
@ -274,10 +276,10 @@ maintainer-clean: maintainer-clean-am
uninstall-e_examplesDATA install-e_examplesDATA tags distdir \
check-TESTS info-am info dvi-am dvi check check-am installcheck-am \
installcheck install-exec-am install-exec install-data-local \
install-data-am install-data install-am install uninstall-am uninstall \
all-redirect all-am all installdirs mostlyclean-generic \
distclean-generic clean-generic maintainer-clean-generic clean \
mostlyclean distclean maintainer-clean
install-data-am install-data install-am install uninstall-local \
uninstall-am uninstall all-redirect all-am all installdirs \
mostlyclean-generic distclean-generic clean-generic \
maintainer-clean-generic clean mostlyclean distclean maintainer-clean
.sh_.sh:
@ -288,12 +290,15 @@ emailrelay: emailrelay.sh
cp emailrelay.sh emailrelay
chmod ugo+x emailrelay
install-data-local:
install-data-local: install-e_examplesDATA install-e_initSCRIPTS
chmod ugo+x $(DESTDIR)$(e_examplesdir)/emailrelay-notify.sh
chmod ugo+x $(DESTDIR)$(e_examplesdir)/emailrelay-resubmit.sh
chmod ugo+x $(DESTDIR)$(e_examplesdir)/emailrelay-deliver.sh
chmod ugo+x $(DESTDIR)$(e_examplesdir)/emailrelay-process.sh
uninstall-local:
-rmdir $(DESTDIR)$(e_examplesdir) 2>/dev/null
# Tell versions [3.59,3.63) of GNU make to not export all variables.
# Otherwise a system limit (for SysV at least) may be exceeded.
.NOEXPORT:

View File

@ -23,6 +23,10 @@
#
awk="gawk"
if test "`echo a | ${awk} '{print}'`" != "a"
then
awk="nawk"
fi 2>/dev/null
# PreFilter()
# Removes banner comments (including legalese) from the top of the file.

View File

@ -59,7 +59,7 @@ Auth()
Content()
{
echo "To:" ${USER}@localhost
echo "To:" ${USER}@`uname -n`
echo "Subject: test message from process" $$
echo "From: tester"
echo ""
@ -75,7 +75,7 @@ Envelope()
echo "X-MailRelay-Content: 8bit"
echo "X-MailRelay-From: me"
echo "X-MailRelay-ToCount: 1"
echo "X-MailRelay-To-Remote:" ${USER}@localhost
echo "X-MailRelay-To-Remote:" ${USER}@`uname -n`
echo "X-MailRelay-Authentication: anon"
echo "X-MailRelay-Client: 127.0.0.1"
echo "X-MailRelay-End: 1"

View File

@ -24,27 +24,201 @@
# A shell-script wrapper for E-MailRelay designed for
# use in the SysV-init system (/etc/init.d).
#
# usage: emailrelay { start [<emailrelay-switches>] | stop }
# See also: LSB, start_daemon (lsb), startproc (suse),
# install_initd (lsb), insserv (suse)
#
# usage: emailrelay { start [<emailrelay-switches>] | stop | restart | force-reload | status }
#
# LSB comment block
#
### BEGIN INIT INFO
# Provides: emailrelay
# Required-Start: $network
# Required-Stop: $network
# Default-Start: 3 4 5
# Default-Stop: 3 4 5
# Description: E-MailRelay store-and-forward Message Transfer Agent.
### END INIT INFO
# choose an infrastructure style
#
if test -f /etc/rc.status
then
style="suse"
. /etc/rc.status
elif test -f /lib/lsb/init-functions
then
style="lsb"
. /lib/lsb/init-functions
else
style="unix"
fi
# configuration
#
switches=""
sbin="/sbin"
var_run="/var/run"
emailrelay="__SBIN_DIR__/emailrelay"
switches=""
# configuration fallback
#
if test \! -d "${var_run}" ; then var_run="/tmp" ; fi
if test \! -x "${emailrelay}" ; then emailrelay="emailrelay" ; fi
# initialisation
#
if test \! -x "${emailrelay}" ; then emailrelay="./emailrelay" ; fi
pid_file="${var_run}/emailrelay.pid"
PATH="${PATH}:/sbin:/bin:/usr/bin"
# functions...
#
# <style>_reset() -- initialise
# <style>_message() -- do echo -n
# <style>_status() [-v] -- save errno and say ok/failed for -v
# <style>_startproc() -- start the server
# <style>_killproc() -- kill the server
# <style>_checkproc() -- return true if running
# <style>_exit() -- exit with saved errno
function unix_reset()
{
unix_errno="0"
}
function unix_message()
{
echo -n "${1}"
}
function unix_status()
{
unix_errno="$?"
if test "${1}" = "-v" -a "${unix_errno}" -eq 0
then
echo " ... done"
fi
if test "${1}" = "-v" -a "${unix_errno}" -ne 0
then
echo " ... failed"
fi
}
function unix_startproc()
{
$@
}
function unix_killproc()
{
if test -f "${pid_file}" && test "`cat ${pid_file}`" != ""
then
kill "`cat ${pid_file}`"
rm -f "${pid_file}" 2>/dev/null
fi
return 0
}
function unix_checkproc()
{
if test -f "${pid_file}" && test "`cat ${pid_file}`" != "" && kill -0 "`cat ${pid_file}`" 2>/dev/null
then
return 0
elif test -f "${pid_file}"
then
return 1
else
return 3
fi
}
function unix_exit()
{
exit "${unix_errno}"
}
##
function suse_reset()
{
rc_reset
}
function suse_message()
{
echo -n "${1}"
}
function suse_status()
{
rc_status $@
}
function suse_startproc()
{
startproc $@
}
function suse_killproc()
{
killproc $@
}
function suse_checkproc()
{
checkproc $@
}
function suse_exit()
{
rc_exit
}
##
function lsb_reset()
{
lsb_errno="0"
lsb_text=""
}
function lsb_message()
{
lsb_text="$@"
}
function lsb_status()
{
lsb_errno=$?
if test "${lsb_errno}" -eq 0 -a "${1}" = "-v"
then
log_success_msg "${lsb_text}: done"
elif test "${1}" = "-v"
then
log_failure_msg "${lsb_text}: failed"
fi
}
function lsb_startproc()
{
start_daemon "`basename \"${1}\"`"
}
function lsb_killproc()
{
killproc "`basename \"${1}\"`"
}
function lsb_checkproc()
{
base="`basename \"${1}\"`"
pids="`pidofproc \"${base}\" | sed 's/ *$//'`"
test "${pids}" != ""
}
function lsb_exit()
{
exit ${lsb_errno}
}
# check the command line
#
usage="{ start | stop }"
usage="{ start [<server-switches>] | stop | restart | force-reload | status }"
if test $# -eq 0
then
echo usage: `basename $0` "${usage}" >&2
@ -53,25 +227,44 @@ fi
# process the command line
#
if test "${1}" = "start"
then
# "start"
#
shift
rm -f "${pid_file}" 2>/dev/null
${emailrelay} --as-server --pid-file "${pid_file}" ${switches} $@
${style}_reset
case "${1}" in
elif test "${1}" = "stop"
then
# "stop"
#
if test -f "${pid_file}" && test "`cat ${pid_file}`" != ""
then
kill "`cat ${pid_file}`"
fi
start)
shift
${style}_message "Starting E-MailRelay server"
${style}_startproc ${emailrelay} --as-server --pid-file "${pid_file}" ${switches} $@
${style}_status -v
;;
else
echo usage: `basename $0` "${usage}" >&2
exit 2
fi
stop)
${style}_message "Shutting down E-MailRelay"
${style}_killproc "${emailrelay}"
${style}_status -v
;;
restart|force-reload)
shift
$0 stop
$0 start $@
${style}_status
;;
reload)
echo usage: `basename $0` reload: not implemented >&2
exit 3
;;
status)
${style}_message "Checking for E-MailRelay"
${style}_checkproc "${emailrelay}"
${style}_status -v
;;
*)
echo usage: `basename $0` "${usage}" >&2
exit 2
;;
esac
${style}_exit

View File

@ -26,15 +26,22 @@
# Only does one sustitution per line. Directives which start the line are
# multi-line. Those within a line are one-line, expanded in-place.
#
# The "-t" switch can be used to suppress expansion of files
# with an extension of ".html".
# The "-t" switch can be used to suppress expansion where the included
# file has an extension of ".html".
#
# Bugs: Does not do nested expansion. Only supports one include statement
# per line.
# In edit mode (--edit) the exit value is 0 (shell true) if the file is
# modified. This allows for recursive expansion using a shell while loop.
#
# usage: expand.sh [-a <awk>] [-t]
# usage: expand.sh [-a <awk>] [-t] { --edit <file> | [<file> ...] }
#
Usage()
{
echo usage: `basename $0` '[-a <awk>] [-t] { --edit <file> | [<file> ...] }' >&2
}
tmp="/tmp/`basename $0`.$$.tmp"
awk="gawk"
if test "${1}" = "-a"
then
@ -50,40 +57,64 @@ then
expand_html="0"
fi
${awk} -v expand_html="${expand_html}" -v cat="${awk} '{print}'" '
{
line = $0
if( match(line,"#include#[^#]*#") )
{
rstart = RSTART
directive = substr(line,RSTART,RLENGTH)
path = substr(line,RSTART+9,RLENGTH-10)
edit="0"
if test "${1}" = "--edit"
then
shift
edit="1"
if test $# -ne 1 ; then Usage ; exit 1 ; fi
fi
if( !expand_html && match(path,".html$") )
Expand()
{
${awk} -v expand_html="${expand_html}" -v cat="${awk} '{print}'" '
BEGIN { modified = 0 }
{
line = $0
if( match(line,"#include#[^#]*#") )
{
print line
}
else
{
head = substr(line,1,rstart-1)
if( match(head,"^[[:space:]]*$") )
rstart = RSTART
directive = substr(line,RSTART,RLENGTH)
path = substr(line,RSTART+9,RLENGTH-10)
if( !expand_html && match(path,".html$") )
{
system( cat " " path )
print line
}
else
{
getline text <path
if( length(text) == 0 )
printf( "expand.sh: warning: line %d: empty text sustitution for %s\n" , NR , path ) >"/dev/fd/2"
sub( directive , text , line )
print line
modified = 1
head = substr(line,1,rstart-1)
if( match(head,"^[[:space:]]*$") )
{
system( cat " " path )
}
else
{
getline text <path
if( length(text) == 0 )
printf( "expand.sh: warning: line %d: empty text sustitution for %s\n" , NR , path ) >"/dev/fd/2"
sub( directive , text , line )
print line
}
}
}
else
{
print line
}
}
else
{
print line
}
END { exit ! modified }
' $@
}
' $@
if test "${edit}" -eq 1
then
# (use cp rather than mv here to protect read-only files)
Expand $@ > "${tmp}" && cp "${tmp}" "${1}"
rc=$? ; rm -f "${tmp}" 2>/dev/null ; exit "${rc}"
else
Expand $@
true
fi

View File

@ -70,7 +70,6 @@ Main()
BEGIN {
if( full )
{
colour = "#ffffff"
dtd_name = "-//W3C//DTD HTML 4.01//EN"
dtd_ref = "http://www.w3.org/TR/html4/strict.dtd"
printf( "<!DOCTYPE HTML PUBLIC \"%s\" \"%s\">\n" , dtd_name , dtd_ref )
@ -81,7 +80,7 @@ Main()
if( length(stylesheet) )
printf( " <link rel=\"stylesheet\" href=\"%s\" type=\"text/css\">\n" , stylesheet )
printf( " </head>\n" )
printf( " <body bgcolor=\"%s\">\n" , colour )
printf( " <body>\n" )
printf( " <!-- index:0::::%s -->\n" , title )
printf( " <div class=\"div-main\">\n" )
}
@ -161,37 +160,37 @@ Main()
# item-outer
else if( match(type,"^item-outer,1[:,]") )
printf( " <ul>\n <li>%s</li>\n" , etail )
printf( " <ul>\n <li>%s\n" , etail )
else if( match(type,"^item-outer[:,]") )
printf( " <li>%s</li>\n" , etail )
printf( " </li>\n <li>%s\n" , etail )
else if( match(type,"^item-outer-end[:,]") )
printf( " </ul>\n" , etail )
printf( " </li>\n </ul>\n" , etail )
# item-inner
else if( match(type,"^item-inner,1[:,]") )
printf( " <ul>\n <li>%s</li>\n" , etail )
printf( " <ul>\n <li>%s</li>\n" , etail )
else if( match(type,"^item-inner[:,]") )
printf( " <li>%s</li>\n" , etail )
printf( " <li>%s</li>\n" , etail )
else if( match(type,"^item-inner-end[:,]") )
printf( " </ul>\n" , etail )
printf( " </ul>\n" , etail )
# item-name
else if( match(type,"^item-name,1[:,]") )
printf( " <dl>\n <dt>%s</dt>\n" , etail )
else if( match(type,"^item-name[:,]") )
printf( " <dt>%s</dt>\n" , etail )
printf( " <dt>%s</dt>\n" , etail )
else if( match(type,"^item-name-end[:,]") )
printf( " </dl>\n" , etail )
# item-detail
else if( match(type,"^item-detail,1[:,]") )
printf( " <dd>\n <p>\n %s\n" , etail )
printf( " <dd>\n %s\n" , etail )
else if( match(type,"^item-detail[:,]") )
printf( " %s\n" , etail )
printf( " %s\n" , etail )
else if( match(type,"^item-detail-end[:,]") )
printf( " </p>\n </dd>\n" )
printf( " </dd>\n" )
else if( match(type,"^item-detail-blank[:,]") )
printf( " </p>\n <p>\n" )
printf( " <p class=\"p-break\"></p>\n" )
# code
else if( match(type,"^code,1[:,]") )
@ -239,7 +238,7 @@ Main()
# image
else if( match(type,"^image[:,]") )
printf( "<img src=\"%s\">\n" , tail )
printf( "<img src=\"%s\" alt=\"%s\">\n" , tail , "image" )
# blank
else if( match(type,"^blank[:,]") )

View File

@ -104,7 +104,7 @@ Main()
is_author = 0
is_html = 0
is_code = match( line , "^" tab )
is_item_outer = match( line , "^+ " )
is_item_outer = match( line , "^\\+ " )
is_item_inner = match( line , "^ - " )
}
@ -119,7 +119,7 @@ Main()
}
else if( is_image )
{
sub( "^<<" , "" , line )
sub( "^[[:space:]]*<<" , "" , line )
sub( ">>[[:space:]]*$" , "" , line )
tagOutputRaw( line , "image" )
}
@ -148,7 +148,7 @@ Main()
}
else if( is_item_outer )
{
sub( "^+ " , "" , line )
sub( "^\\+ " , "" , line )
tagOutput( line , "item-outer" )
}
else if( is_item_inner )
@ -213,14 +213,15 @@ Main()
#
Number()
{
${awk} -v item_tag="${1}" -v ignore_1="${2}" -v ignore_2="${3}" -v ignore_3="${4}" '
${awk} -v item_tag="${1}" -v ignore_1="${2}" -v ignore_2="${3}" -v ignore_3="${4}" -v ignore_4="${5}" '
function ignore_line( line )
{
i_0 = match( line , "^ignore" )
i_1 = length(ignore_1) && match( line , "^" ignore_1 "[:,]" )
i_2 = length(ignore_2) && match( line , "^" ignore_2 "[:,]" )
i_3 = length(ignore_3) && match( line , "^" ignore_3 "[:,]" )
return i_0 || i_1 || i_2 || i_3
i_4 = length(ignore_3) && match( line , "^" ignore_4 "[:,]" )
return i_0 || i_1 || i_2 || i_3 || i_4
}
BEGIN {
n = 1
@ -327,8 +328,8 @@ Cat ${file} | \
Number "item-outer" "item-inner" "blank" | \
Number "item-inner" "blank" | \
Number "item-numbered" | \
Number "item-name" "item-detail" "blank" "item-detail-blank" | \
Number "item-detail" "item-detail-blank" | \
Number "item-name" "item-detail" "blank" "item-detail-blank" "code" | \
Number "item-detail" "item-detail-blank" "code" | \
Number "code" "blank" "image" | \
Number "footer" "blank" "image" | \
Number "citation" "blank" | \

View File

@ -6,15 +6,18 @@
/* Define if you have the `glob' function. */
#undef HAVE_GLOB
/* have reentrant gmtime */
/* Define if gmtime_r in time.h */
#undef HAVE_GMTIME_R
/* have reentrant localtime */
/* Define if localtime_r in time.h */
#undef HAVE_LOCALTIME_R
/* Define if you have the <ndir.h> header file, and it defines `DIR'. */
#undef HAVE_NDIR_H
/* Define if socklen_t type definition in sys/socket.h */
#undef HAVE_SOCKLEN_T
/* Define if you have the <sys/dir.h> header file, and it defines `DIR'. */
#undef HAVE_SYS_DIR_H

978
configure vendored

File diff suppressed because it is too large Load Diff

15
configure.ac Executable file → Normal file
View File

@ -20,8 +20,8 @@ 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.9)
AC_INIT(src/gsmtp/gsmtp.h)
AM_INIT_AUTOMAKE(emailrelay,1.0.0)
AM_CONFIG_HEADER(config.h)
dnl ===
@ -37,6 +37,7 @@ AC_CHECK_PROGS(AR,ar gar)
AC_CHECK_PROGS(GZIP,gzip)
AC_CHECK_PROG(HAVE_DOXYGEN,doxygen,yes)
AC_CHECK_PROG(HAVE_MAN2HTML,man2html,yes)
ACLOCAL_COMPILER_VERSION
dnl ===
dnl check for libraries...
@ -55,11 +56,9 @@ AC_CHECK_HEADERS(unistd.h)
AC_CHECK_HEADERS(sys/time.h)
AC_CHECK_FUNCS(glob)
AC_LANG_CPLUSPLUS
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)
AC_EGREP_HEADER(localtime_r,time.h,[AC_DEFINE(HAVE_LOCALTIME_R,1,[have reentrant localtime])])
AC_EGREP_HEADER(gmtime_r,time.h,[AC_DEFINE(HAVE_GMTIME_R,1,[have reentrant gmtime])])
ACLOCAL_TYPE_SOCKLEN_T
ACLOCAL_CHECK_GMTIME_R
ACLOCAL_CHECK_LOCALTIME_R
dnl ===
dnl directory tweaking...
@ -89,5 +88,5 @@ SET_MAKE=""
dnl ===
dnl generate files...
dnl
AC_OUTPUT(Makefile src/Makefile src/glib/Makefile src/gnet/Makefile src/main/Makefile src/win32/Makefile lib/Makefile lib/gcc2.95/Makefile lib/msvc6.0/Makefile bin/Makefile doc/Makefile)
AC_OUTPUT(Makefile src/Makefile src/glib/Makefile src/gnet/Makefile src/gsmtp/Makefile src/main/Makefile src/win32/Makefile lib/Makefile lib/gcc2.95/Makefile lib/msvc6.0/Makefile bin/Makefile doc/Makefile)

View File

@ -24,8 +24,9 @@ man_files_out=emailrelay.1.gz emailrelay-passwd.1.gz emailrelay-poke.1.gz emailr
html_files_in=doxygen_header.html
html_files_thru=index.html $(stylesheet)
html_files_out=readme.html developer.html reference.html userguide.html windows.html emailrelay-man.html changelog.html
png_files=gsmtp-classes.png gnet-classes.png
EXTRA_DIST = $(man_files_in) $(txt_files) $(html_files_in) $(html_files_thru)
EXTRA_DIST = $(man_files_in) $(txt_files) $(html_files_in) $(html_files_thru) $(png_files)
noinst_SCRIPTS = .dox
e_man1_DATA = $(man_files_out)
e_doc_DATA = $(txt_files) $(html_files_out) $(html_files_thru)
@ -45,7 +46,7 @@ converter_helper3=$(top_builddir)/bin/expand.sh
converter_helper3_src=$(top_srcdir)/bin/expand.sh_
.txt.html:
$(converter) -a "$(AWK)" $< $(stylesheet) > $*.html
$(converter) -a "$(AWK)" $(top_srcdir)/doc/$*.txt $(stylesheet) > $*.html
$(filter): $(filter_src)
cp $(filter_src) $(filter)
@ -71,7 +72,7 @@ $(converter_helper3): $(converter_helper3_src)
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
emailrelay-man.html: emailrelay.1
if test "$(HAVE_MAN2HTML)" = "yes" ; then man2html $< > emailrelay-man.html ; fi
if test "$(HAVE_MAN2HTML)" = "yes" ; then man2html emailrelay.1 > emailrelay-man.html ; fi
developer.html reference.html userguide.html: $(converter)
@ -82,21 +83,22 @@ changelog.html: $(top_srcdir)/ChangeLog $(converter)
$(converter) -a "$(AWK)" $(top_srcdir)/ChangeLog $(stylesheet) > changelog.html
emailrelay.1.gz : emailrelay.1
if test -n "$(GZIP)" ; then $(GZIP) -c $< > emailrelay.1.gz ; fi
if test -n "$(GZIP)" ; then $(GZIP) -c $(top_srcdir)/doc/emailrelay.1 > emailrelay.1.gz ; fi
emailrelay-passwd.1.gz : emailrelay-passwd.1
if test -n "$(GZIP)" ; then $(GZIP) -c $< > emailrelay-passwd.1.gz ; fi
if test -n "$(GZIP)" ; then $(GZIP) -c $(top_srcdir)/doc/emailrelay-passwd.1 > emailrelay-passwd.1.gz ; fi
emailrelay-submit.1.gz : emailrelay-submit.1
if test -n "$(GZIP)" ; then $(GZIP) -c $< > emailrelay-submit.1.gz ; fi
if test -n "$(GZIP)" ; then $(GZIP) -c $(top_srcdir)/doc/emailrelay-submit.1 > emailrelay-submit.1.gz ; fi
emailrelay-poke.1.gz: emailrelay-poke.1
if test -n "$(GZIP)" ; then $(GZIP) -c $< > emailrelay-poke.1.gz ; fi
if test -n "$(GZIP)" ; then $(GZIP) -c $(top_srcdir)/doc/emailrelay-poke.1 > emailrelay-poke.1.gz ; fi
install-data-local:
install-data-local: install-e_docDATA install-e_man1DATA
$(mkinstalldirs) $(DESTDIR)$(e_docdir)/doxygen
if test "$(HAVE_DOXYGEN)" = "yes" ; then for file in doxygen/* ; do $(INSTALL) $$file $(DESTDIR)$(e_docdir)/$$file ; done ; fi
uninstall-local:
-rm -f $(DESTDIR)$(e_docdir)/html/* 2>/dev/null
-rmdir $(DESTDIR)$(e_docdir)/doxygen 2>/dev/null
-rmdir $(DESTDIR)$(e_docdir) 2>/dev/null

View File

@ -79,6 +79,7 @@ POST_UNINSTALL = :
AR = @AR@
AWK = @AWK@
CC = @CC@
COMPILER_VERSION = @COMPILER_VERSION@
CXX = @CXX@
GZIP = @GZIP@
HAVE_DOXYGEN = @HAVE_DOXYGEN@
@ -103,8 +104,9 @@ man_files_out = emailrelay.1.gz emailrelay-passwd.1.gz emailrelay-poke.1.gz emai
html_files_in = doxygen_header.html
html_files_thru = index.html $(stylesheet)
html_files_out = readme.html developer.html reference.html userguide.html windows.html emailrelay-man.html changelog.html
png_files = gsmtp-classes.png gnet-classes.png
EXTRA_DIST = $(man_files_in) $(txt_files) $(html_files_in) $(html_files_thru)
EXTRA_DIST = $(man_files_in) $(txt_files) $(html_files_in) $(html_files_thru) $(png_files)
noinst_SCRIPTS = .dox
e_man1_DATA = $(man_files_out)
e_doc_DATA = $(txt_files) $(html_files_out) $(html_files_thru)
@ -272,7 +274,7 @@ maintainer-clean-generic clean mostlyclean distclean maintainer-clean
.txt.html:
$(converter) -a "$(AWK)" $< $(stylesheet) > $*.html
$(converter) -a "$(AWK)" $(top_srcdir)/doc/$*.txt $(stylesheet) > $*.html
$(filter): $(filter_src)
cp $(filter_src) $(filter)
@ -298,7 +300,7 @@ $(converter_helper3): $(converter_helper3_src)
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
emailrelay-man.html: emailrelay.1
if test "$(HAVE_MAN2HTML)" = "yes" ; then man2html $< > emailrelay-man.html ; fi
if test "$(HAVE_MAN2HTML)" = "yes" ; then man2html emailrelay.1 > emailrelay-man.html ; fi
developer.html reference.html userguide.html: $(converter)
@ -309,23 +311,24 @@ changelog.html: $(top_srcdir)/ChangeLog $(converter)
$(converter) -a "$(AWK)" $(top_srcdir)/ChangeLog $(stylesheet) > changelog.html
emailrelay.1.gz : emailrelay.1
if test -n "$(GZIP)" ; then $(GZIP) -c $< > emailrelay.1.gz ; fi
if test -n "$(GZIP)" ; then $(GZIP) -c $(top_srcdir)/doc/emailrelay.1 > emailrelay.1.gz ; fi
emailrelay-passwd.1.gz : emailrelay-passwd.1
if test -n "$(GZIP)" ; then $(GZIP) -c $< > emailrelay-passwd.1.gz ; fi
if test -n "$(GZIP)" ; then $(GZIP) -c $(top_srcdir)/doc/emailrelay-passwd.1 > emailrelay-passwd.1.gz ; fi
emailrelay-submit.1.gz : emailrelay-submit.1
if test -n "$(GZIP)" ; then $(GZIP) -c $< > emailrelay-submit.1.gz ; fi
if test -n "$(GZIP)" ; then $(GZIP) -c $(top_srcdir)/doc/emailrelay-submit.1 > emailrelay-submit.1.gz ; fi
emailrelay-poke.1.gz: emailrelay-poke.1
if test -n "$(GZIP)" ; then $(GZIP) -c $< > emailrelay-poke.1.gz ; fi
if test -n "$(GZIP)" ; then $(GZIP) -c $(top_srcdir)/doc/emailrelay-poke.1 > emailrelay-poke.1.gz ; fi
install-data-local:
install-data-local: install-e_docDATA install-e_man1DATA
$(mkinstalldirs) $(DESTDIR)$(e_docdir)/doxygen
if test "$(HAVE_DOXYGEN)" = "yes" ; then for file in doxygen/* ; do $(INSTALL) $$file $(DESTDIR)$(e_docdir)/$$file ; done ; fi
uninstall-local:
-rm -f $(DESTDIR)$(e_docdir)/html/* 2>/dev/null
-rmdir $(DESTDIR)$(e_docdir)/doxygen 2>/dev/null
-rmdir $(DESTDIR)$(e_docdir) 2>/dev/null
# Tell versions [3.59,3.63) of GNU make to not export all variables.
# Otherwise a system limit (for SysV at least) may be exceeded.

View File

@ -3,15 +3,12 @@ E-MailRelay Developer guide
Module structure
----------------
There are two C++ libraries in the E-MailRelay code: "glib" provides low-level
There are three C++ libraries in the E-MailRelay code: "glib" provides low-level
classes for file-system abstraction, date and time representation, string
utility functions, logging, command line parsing etc., and "gnet" provides
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" and "Main"
namespaces. The key interfaces in the "GSmtp" namespace are "ClientProtocol",
"ServerProtocol" and "MessageStore".
utility functions, logging, command line parsing etc., "gnet" provides network
classes using the Berkley socket and Winsock APIs, and "gsmtp" contains SMTP and
message-store classes. All three libraries are portable between POSIX-like
systems (eg. Linux) and Windows.
Under Windows there is an additional library for event handling. Windows has
historically built network event processing on top of the GUI event system,
@ -38,8 +35,8 @@ another for immediate forwarding ("ProtocolMessageForward").
The protocol and message-store functionality are brought together by the
high-level "GSmtp::Server" and "GSmtp::Client" classes.
Simplified class diagrams for the *GNet* [graphics/gnet-classes.png] and
*GSmtp* [graphics/gsmtp-classes.png] namespaces are available.
Simplified class diagrams for the *GNet* [gnet-classes.png] and
*GSmtp* [gsmtp-classes.png] namespaces are available.
Directory structure
-------------------
@ -57,6 +54,10 @@ Directory structure
A network library using Berkley sockets or Winsock.
# src/gsmtp
An SMTP library.
# src/win32
Additional classes for windows event processing.
@ -87,9 +88,9 @@ For a quick bottom-up tour of the code take a look at the following headers:
* src/gnet/gaddress.h
* src/gnet/gsocket.h
* src/gnet/gserver.h
* src/main/gmessagestore.h
* src/main/gserverprotocol.h
* src/main/gsmtpserver.h
* src/smtp/gmessagestore.h
* src/smtp/gserverprotocol.h
* src/smtp/gsmtpserver.h
Portability
-----------

View File

@ -58,13 +58,13 @@ server.
Enables the administration interface and specifies its listening port number.
.TP
.B \-q,--as-client \fIhost:port\fR
Equivalent to \fI--log\fR \fI--no-syslog\fR \fI--no-daemon\fR \fI--dont-serve\fR \fI--forward\fR \fI--forward-to\fR.
Runs as a client, forwarding spooled mail to <host>: equivalent to \fI--log\fR \fI--no-syslog\fR \fI--no-daemon\fR \fI--dont-serve\fR \fI--forward\fR \fI--forward-to\fR.
.TP
.B \-y,--as-proxy \fIhost:port\fR
Equivalent to \fI--log\fR \fI--close-stderr\fR \fI--immediate\fR \fI--forward-to\fR.
Runs as a proxy: equivalent to \fI--log\fR \fI--close-stderr\fR \fI--immediate\fR \fI--forward-to\fR.
.TP
.B \-d,--as-server
Equivalent to \fI--log\fR \fI--close-stderr\fR.
Runs as a server: equivalent to \fI--log\fR \fI--close-stderr\fR.
.TP
.B \-C,--client-auth \fIfile\fR
Enables authentication with remote server, using the given secrets file.
@ -75,14 +75,20 @@ Closes the standard error stream after start-up.
.B \-U,--connection-timeout \fItime\fR
Sets the timeout (in seconds) when connecting to a remote server (default is 40).
.TP
.B \-g,--debug
Generates debug-level logging (if compiled-in).
.TP
.B \-D,--domain \fIfqdn\fR
Sets an override for the host's fully qualified domain name.
.TP
.B \-X,--dont-listen
Dont listen for smtp connections (usually used with \fI--admin\fR).
.TP
.B \-x,--dont-serve
Stops the process acting as a server (usually used with \fI--forward\fR).
Dont act as a server (usually used with \fI--forward\fR).
.TP
.B \-z,--filter \fIprogram\fR
Defines a mail pre-processor.
Defines a mail processor program.
.TP
.B \-f,--forward
Forwards stored mail on startup (requires \fI--forward-to\fR).
@ -96,8 +102,11 @@ Displays help text and exits.
.B \-m,--immediate
Forwards each message as soon as it is received (requires \fI--forward-to\fR).
.TP
.B \-I,--interface \fIip-address\fR
Listen on a specific interface.
.TP
.B \-l,--log
Writes log information on standard error (if open) and syslog (if not disabled).
Writes log information on standard error and syslog.
.TP
.B \-L,--log-time
Adds a timestamp to the logging output.
@ -124,68 +133,33 @@ Sets the response timeout (in seconds) when talking to a remote server (default
Enables authentication of remote clients, using the given secrets file.
.TP
.B \-s,--spool-dir \fIdir\fR
Specifies the spool directory (default is \fI/usr/local/var/spool/emailrelay\fR).
Specifies the spool directory (default is \fI/var/spool/emailrelay\fR).
.TP
.B \-u,--user \fIusername\fR
Names the effective user to switch to when started as root (default is \fIdaemon\fR).
.TP
.B \-v,--verbose
Generates more verbose logging (if compiled-in and logging enabled and stderr open).
Generates more verbose output (works with \fI--help\fR and \fI--log\fR).
.TP
.B \-Z,--verifier \fIprogram\fR
Defines an external address verifier program.
.TP
.B \-V,--version
Displays version information and exits.
.SH FILES
GNU style...
.br
/usr/local/libexec/emailrelay-passwd
.br
/usr/local/libexec/emailrelay-poke
.br
/usr/local/libexec/emailrelay
.br
/usr/local/libexec/emailrelay-notify.sh
.br
/usr/local/libexec/emailrelay-deliver.sh
.br
/usr/local/libexec/emailrelay-notify.sh
.br
/usr/local/libexec/emailrelay-poke
.br
/usr/local/libexec/emailrelay-process.sh
.br
/usr/local/sbin/emailrelay-submit
.br
/usr/local/sbin/emailrelay
.br
/usr/local/share/emailrelay/doc/developer.txt
.br
/usr/local/share/emailrelay/doc/reference.txt
.br
/usr/local/share/emailrelay/doc/userguide.txt
.br
/usr/local/share/emailrelay/doc/windows.txt
.br
/usr/local/share/emailrelay/doc/readme.html
.br
/usr/local/share/emailrelay/doc/developer.html
.br
/usr/local/share/emailrelay/doc/reference.html
.br
/usr/local/share/emailrelay/doc/userguide.html
.br
/usr/local/share/emailrelay/doc/windows.html
.br
/usr/local/share/emailrelay/doc/emailrelay-man.html
.br
/usr/local/share/emailrelay/doc/index.html
.br
/usr/local/share/emailrelay/doc/changelog.html
.br
/usr/local/share/emailrelay/doc/emailrelay.css
.br
/usr/local/share/emailrelay/doc/NEWS
.br
/usr/local/share/emailrelay/doc/README
.br
/usr/local/share/emailrelay/doc/changelog.gz
.br
/usr/local/man/man1/emailrelay.1.gz
/usr/local/libexec/emailrelay-resubmit.sh
.br
/usr/local/man/man1/emailrelay-passwd.1.gz
.br
@ -193,51 +167,59 @@ GNU style...
.br
/usr/local/man/man1/emailrelay-submit.1.gz
.br
/usr/local/var/spool/emailrelay/emailrelay.*.envelope
/usr/local/man/man1/emailrelay.1.gz
.br
/usr/local/sbin/emailrelay
.br
/usr/local/sbin/emailrelay-passwd
.br
/usr/local/sbin/emailrelay-submit
.br
/usr/local/share/emailrelay/doc/NEWS
.br
/usr/local/share/emailrelay/doc/README
.br
/usr/local/share/emailrelay/doc/changelog.gz
.br
/usr/local/share/emailrelay/doc/changelog.html
.br
/usr/local/share/emailrelay/doc/developer.html
.br
/usr/local/share/emailrelay/doc/developer.txt
.br
/usr/local/share/emailrelay/doc/emailrelay.css
.br
/usr/local/share/emailrelay/doc/index.html
.br
/usr/local/share/emailrelay/doc/readme.html
.br
/usr/local/share/emailrelay/doc/reference.html
.br
/usr/local/share/emailrelay/doc/reference.txt
.br
/usr/local/share/emailrelay/doc/userguide.html
.br
/usr/local/share/emailrelay/doc/userguide.txt
.br
/usr/local/share/emailrelay/doc/windows.html
.br
/usr/local/share/emailrelay/doc/windows.txt
.br
/usr/local/var/spool/emailrelay/emailrelay.*.content
.br
/usr/local/var/spool/emailrelay/emailrelay.*.envelope
.LP
FHS style...
.br
/usr/lib/emailrelay-passwd
/etc/init.d/emailrelay
.br
/usr/lib/emailrelay-poke
.br
/usr/sbin/emailrelay-submit
/usr/lib/emailrelay/emailrelay-poke
.br
/usr/sbin/emailrelay
.br
/usr/share/doc/emailrelay/examples/emailrelay-notify.sh
/usr/sbin/emailrelay-passwd
.br
/usr/share/doc/emailrelay/examples/emailrelay-deliver.sh
.br
/usr/share/doc/emailrelay/examples/emailrelay-process.sh
.br
/usr/share/doc/emailrelay/developer.txt
.br
/usr/share/doc/emailrelay/reference.txt
.br
/usr/share/doc/emailrelay/userguide.txt
.br
/usr/share/doc/emailrelay/windows.txt
.br
/usr/share/doc/emailrelay/readme.html
.br
/usr/share/doc/emailrelay/developer.html
.br
/usr/share/doc/emailrelay/reference.html
.br
/usr/share/doc/emailrelay/userguide.html
.br
/usr/share/doc/emailrelay/windows.html
.br
/usr/share/doc/emailrelay/emailrelay-man.html
.br
/usr/share/doc/emailrelay/index.html
.br
/usr/share/doc/emailrelay/changelog.html
.br
/usr/share/doc/emailrelay/emailrelay.css
/usr/sbin/emailrelay-submit
.br
/usr/share/doc/emailrelay/NEWS
.br
@ -245,7 +227,37 @@ FHS style...
.br
/usr/share/doc/emailrelay/changelog.gz
.br
/usr/share/man/man1/emailrelay.1.gz
/usr/share/doc/emailrelay/changelog.html
.br
/usr/share/doc/emailrelay/developer.html
.br
/usr/share/doc/emailrelay/developer.txt
.br
/usr/share/doc/emailrelay/emailrelay.css
.br
/usr/share/doc/emailrelay/examples/emailrelay-deliver.sh
.br
/usr/share/doc/emailrelay/examples/emailrelay-notify.sh
.br
/usr/share/doc/emailrelay/examples/emailrelay-process.sh
.br
/usr/share/doc/emailrelay/examples/emailrelay-resubmit.sh
.br
/usr/share/doc/emailrelay/index.html
.br
/usr/share/doc/emailrelay/readme.html
.br
/usr/share/doc/emailrelay/reference.html
.br
/usr/share/doc/emailrelay/reference.txt
.br
/usr/share/doc/emailrelay/userguide.html
.br
/usr/share/doc/emailrelay/userguide.txt
.br
/usr/share/doc/emailrelay/windows.html
.br
/usr/share/doc/emailrelay/windows.txt
.br
/usr/share/man/man1/emailrelay-passwd.1.gz
.br
@ -253,11 +265,11 @@ FHS style...
.br
/usr/share/man/man1/emailrelay-submit.1.gz
.br
/etc/init.d/emailrelay
.br
/var/spool/emailrelay/emailrelay.*.envelope
/usr/share/man/man1/emailrelay.1.gz
.br
/var/spool/emailrelay/emailrelay.*.content
.br
/var/spool/emailrelay/emailrelay.*.envelope
.SH SEE ALSO
E-MailRelay user guide
.br

View File

@ -8,6 +8,7 @@ div.two-column-content
div.two-column-menu
{
color: inherit ;
background-color: #eee ;
position: absolute ;
@ -28,7 +29,9 @@ div.two-column-menu
body
{
font-family: verdana, arial, helvetica, sans-serif ;
background-color: white ;
color: #000 ;
background-color: #fff ;
}
div.div-main
@ -39,7 +42,8 @@ div.div-main
div.div-toc
{
background: #eee ;
color: inherit ;
background-color: #eee ;
}
p.p-toc
@ -55,7 +59,8 @@ div.div-pre
{
margin-left: 3% ;
/* font-size: smaller ; */ /* smaller is unreadable on ie5 */
background: #eee ;
color: inherit ;
background-color: #eee ;
padding: 0.5em ;
border: none ;
}
@ -64,6 +69,7 @@ h1
{
text-align: center ;
color: #09c ;
background-color: inherit ;
}
img
@ -74,22 +80,25 @@ img
h2
{
color: #09c ;
background-color: inherit ;
}
p
{
font: verdana, arial, helvetica, sans-serif ;
font-family: verdana, arial, helvetica, sans-serif ;
text-align: justify ;
}
a.a-toc
{
color: #09c ;
background-color: inherit ;
text-decoration: none ;
}
a.a-toc:hover
{
color: #09c ;
background-color: #ccc ;
}
@ -107,20 +116,24 @@ a.a-href
a.a-href:link
{
color: #09c ;
background-color: inherit ;
}
a.a-href:visited
{
color: #07a ;
background-color: inherit ;
}
a.a-href:hover
{
color: #09c ;
background-color: #eee ;
}
div.two-column-header
{
color: inherit ;
background-color: #eee ;
border-style: solid ;

BIN
doc/gnet-classes.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 16 KiB

BIN
doc/gsmtp-classes.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 10 KiB

View File

@ -8,15 +8,15 @@
<div class="div-main">
<h1>E-MailRelay Documentation</h1>
<ul>
<li><a href="readme.html">Readme</a></li>
<li><a href="changelog.html">Change log</a></li>
<li><a href="userguide.html">User guide</a></li>
<li><a href="reference.html">Reference manual</a></li>
<li><a href="windows.html">Windows installation guide</a></li>
<li><a href="developer.html">Notes for developers</a></li>
<li><a href="doxygen/index.html">Source code documentation</a> (generated by <a href="http://www.doxygen.org">doxygen</a>, if available)</li>
<li><a href="emailrelay-man.html">Man page</a> (generated by man2html, if available)</li>
<li><a href="http://emailrelay.sourceforge.net">Web site</a></li>
<li><a class="a-href" href="readme.html">Readme</a></li>
<li><a class="a-href" href="changelog.html">Change log</a></li>
<li><a class="a-href" href="userguide.html">User guide</a></li>
<li><a class="a-href" href="reference.html">Reference manual</a></li>
<li><a class="a-href" href="windows.html">Windows installation guide</a></li>
<li><a class="a-href" href="developer.html">Notes for developers</a></li>
<li><a class="a-href" href="doxygen/index.html">Source code documentation</a> (generated by <a class="a-href" href="http://www.doxygen.org">doxygen</a>, if available)</li>
<li><a class="a-href" href="emailrelay-man.html">Man page</a> (generated by man2html, if available)</li>
<li><a class="a-href" href="http://emailrelay.sourceforge.net">Web site</a></li>
</ul>
</div>
</body>

View File

@ -13,18 +13,17 @@ The "emailrelay" program supports the following command-line usage:
emailrelay [<switch> [<switch> ...]]
where <switch> is:
# --admin (-a)
Enables the administration interface and specifies its listening port number.
# --as-client (-q)
Equivalent to "--log --no-syslog --no-daemon --dont-serve --forward --forward-to".
Runs as a client, forwarding spooled mail to <host>: equivalent to "--log --no-syslog --no-daemon --dont-serve --forward --forward-to".
# --as-proxy (-y)
Equivalent to "--log --close-stderr --immediate --forward-to".
Runs as a proxy: equivalent to "--log --close-stderr --immediate --forward-to".
# --as-server (-d)
Equivalent to "--log --close-stderr".
Runs as a server: equivalent to "--log --close-stderr".
# --client-auth (-C)
Enables authentication with remote server, using the given secrets file.
@ -35,14 +34,20 @@ where <switch> is:
# --connection-timeout (-U)
Sets the timeout (in seconds) when connecting to a remote server (default is 40).
# --debug (-g)
Generates debug-level logging (if compiled-in).
# --domain (-D)
Sets an override for the host's fully qualified domain name.
# --dont-listen (-X)
Dont listen for smtp connections (usually used with --admin).
# --dont-serve (-x)
Stops the process acting as a server (usually used with --forward).
Dont act as a server (usually used with --forward).
# --filter (-z)
Defines a mail pre-processor.
Defines a mail processor program.
# --forward (-f)
Forwards stored mail on startup (requires --forward-to).
@ -56,8 +61,11 @@ where <switch> is:
# --immediate (-m)
Forwards each message as soon as it is received (requires --forward-to).
# --interface (-I)
Listen on a specific interface.
# --log (-l)
Writes log information on standard error (if open) and syslog (if not disabled).
Writes log information on standard error and syslog.
# --log-time (-L)
Adds a timestamp to the logging output.
@ -84,13 +92,16 @@ where <switch> is:
Enables authentication of remote clients, using the given secrets file.
# --spool-dir (-s)
Specifies the spool directory (default is "/usr/local/var/spool/emailrelay").
Specifies the spool directory (default is "/var/spool/emailrelay").
# --user (-u)
Names the effective user to switch to when started as root (default is "daemon").
# --verbose (-v)
Generates more verbose logging (if compiled-in and logging enabled and stderr open).
Generates more verbose output (works with --help and --log).
# --verifier (-Z)
Defines an external address verifier program.
# --version (-V)
Displays version information and exits.
@ -181,6 +192,29 @@ SMTP issues
re-encode 8-bit messages into 7-bit messages if the next-hop server
does not.
Address validation
------------------
An address supplied to the SMTP commands "RCPT" and "VRFY" is normally accepted
if it contains an at sign, or if it is some sort of local "postmaster" address
(as described above). However, this behaviour can be modified by using an external
verifier program, using the "--verifier" command-line switch.
The verifier program is passed a command-line containing the full address, the
user-name part of the address, the host-name part, the local host's fully
qualified domain name, and the current "MAIL" command's "FROM:" address or the
empty string for the "VRFY" command.
For valid local mailbox addresses the verifier is expected to write two lines to
the standard output -- the full name associated with the mailbox, and the
canonical mailbox name -- and then exit with a value of zero. For valid non-local
addresses the first line of output should be empty, the second line should be
copied from the first command-line argument, and the exit value should be one.
For invalid addresses the exit value should be greater than one, and anything
written to the standard output is taken as the failure reason.
(Only the few few thousand characters are read from the verifier's standard
output stream; any more is thrown away.)
Administration interface
------------------------
If enabled, the server will provide a network interface for performing
@ -198,31 +232,31 @@ functionality of storage daemon and forwarding client. This might be useful
in systems which are low on memory: the memory footprint of "telnet" or
"emailrelay-poke" will be much smaller than "emailrelay --as-client".
Mail pre-processing
-------------------
The "--filter" command-line switch allows you to specify a pre-processor program
Mail processing
---------------
The "--filter" command-line switch allows you to specify a mail 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
mail 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
personal mail 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):
For example, the following command will start a proxy server on port 10025 which
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
The pre-processor program should terminate with an exit code of zero to indicate
The mail processor program should terminate with an exit code of zero to indicate
success. An exit code of 100 can be used to cancel all further processing of the
message, and any other non-zero exit code is used to indicate an error.
For example, the following pre-processor shell script examines the client's
For example, the following mail processor shell script examines the client's
IP address and conditionally dumps the message into "sendmail" (using the
sendmail command-line interface rather than SMTP):
@ -239,14 +273,14 @@ sendmail command-line interface rather than SMTP):
fi
exit 0
An example pre-processor script which does simple rot-13 masking of messages is
An example mail processor script which does simple rot-13 masking of messages is
also provided in the distribution ("share/emailrelay/emailrelay-process.sh").
This could be used as a template for a more sophisticated message encryption
system.
Security issues
---------------
A major security concern is the use of an external mail pre-processor (using the
A major security concern is the use of an external mail processor (using the
"--filter" switch), and so the following precautions are taken:
# effective userid
@ -261,13 +295,13 @@ A major security concern is the use of an external mail pre-processor (using the
# execution environment
The pre-processor runs with an almost empty set of environment variables ("PATH"
The mail processor runs with an almost empty set of environment variables ("PATH"
and "IFS"), and with no open file descriptors other than "stdin"/"stdout"/"stderr"
open onto "/dev/null".
# configuration
The pre-processor filename has to be configured using a full path, so there is
The mail processor filename has to be configured using a full path, so there is
no dependence on the current working directory or the PATH environment variable.
Security issues which relate to the SMTP protocol itself are beyond the scope of
@ -284,7 +318,7 @@ Some other points are:
* 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.
* No exec(), system() or popen() calls are used other than execve() to spawn the mail processor.
* The submit utility is installed as set-group-id with group ownership of "daemon".
Authentication
@ -339,7 +373,7 @@ least make sure that the secrets file has tight permissions, and that the
passwords in it are not also used for anything important (such as root access).
On the server side authentication is advertised in the response to the SMTP
"EHLO" command if the "--auth-server" command line switch is used, but
"EHLO" command if the "--auth-server" command-line switch is used, but
authentication by the client is optional. If the client does authenticate then
the authenticated user-id is stored with the message and then passed on to a
next-hop server using an "AUTH=userid" parameter on the SMTP "MAIL FROM"
@ -362,80 +396,99 @@ from a particular IP address is allowed. This type of POP-before-SMTP
authentication can be done outside the E-MailRelay system by POP/IMAP utilities
such as "fetchmail".
Installation directories
------------------------
Application notes
-----------------
# *Mutt* [http://www.mutt.org]
Mutt can be configured to use the E-MailRelay 'submit' program rather than
"sendmail" with an entry in the ".muttrc" file like this:
set sendmail="/usr/local/libexec/emailrelay-submit"
# *Spam Assassin* [http://spamassassin.taint.org]
The E-MailRelay server can use Spam Assasin to mark potential spam by having
a small shell script which calls spamassassin's "spamc" program installed
as the E-MailRelay mail processor (using the "--filter" command-line switch).
The shell script would look something like this:
#!/bin/sh
tmp="`basename $0`.$$.tmp"
cat ${1} | spamc > ${tmp} && mv ${tmp} ${1}
Files and directories
---------------------
By default "make install" installs files in the following locations:
* /usr/local/libexec/emailrelay-poke
* /usr/local/libexec/emailrelay-notify.sh
* /usr/local/libexec/emailrelay-deliver.sh
* /usr/local/libexec/emailrelay-resubmit.sh
* /usr/local/libexec/emailrelay
* /usr/local/libexec/emailrelay-deliver.sh
* /usr/local/libexec/emailrelay-notify.sh
* /usr/local/libexec/emailrelay-poke
* /usr/local/libexec/emailrelay-process.sh
* /usr/local/sbin/emailrelay
* /usr/local/sbin/emailrelay-passwd
* /usr/local/sbin/emailrelay-submit
* /usr/local/var/spool/emailrelay/*
* /usr/local/share/emailrelay/doc/developer.txt
* /usr/local/share/emailrelay/doc/reference.txt
* /usr/local/share/emailrelay/doc/userguide.txt
* /usr/local/share/emailrelay/doc/windows.txt
* /usr/local/share/emailrelay/doc/readme.html
* /usr/local/share/emailrelay/doc/developer.html
* /usr/local/share/emailrelay/doc/reference.html
* /usr/local/share/emailrelay/doc/userguide.html
* /usr/local/share/emailrelay/doc/windows.html
* /usr/local/share/emailrelay/doc/emailrelay-man.html
* /usr/local/share/emailrelay/doc/index.html
* /usr/local/share/emailrelay/doc/changelog.html
* /usr/local/share/emailrelay/doc/emailrelay.css
* /usr/local/share/emailrelay/doc/NEWS
* /usr/local/share/emailrelay/doc/README
* /usr/local/share/emailrelay/doc/changelog.gz
* /usr/local/share/emailrelay/doc/doxygen/*
* /usr/local/man/man1/emailrelay.1.gz
* /usr/local/libexec/emailrelay-resubmit.sh
* /usr/local/man/man1/emailrelay-passwd.1.gz
* /usr/local/man/man1/emailrelay-poke.1.gz
* /usr/local/man/man1/emailrelay-submit.1.gz
* /usr/local/man/man1/emailrelay.1.gz
* /usr/local/sbin/emailrelay
* /usr/local/sbin/emailrelay-passwd
* /usr/local/sbin/emailrelay-submit
* /usr/local/share/emailrelay/doc/NEWS
* /usr/local/share/emailrelay/doc/README
* /usr/local/share/emailrelay/doc/changelog.gz
* /usr/local/share/emailrelay/doc/changelog.html
* /usr/local/share/emailrelay/doc/developer.html
* /usr/local/share/emailrelay/doc/developer.txt
* /usr/local/share/emailrelay/doc/emailrelay.css
* /usr/local/share/emailrelay/doc/index.html
* /usr/local/share/emailrelay/doc/readme.html
* /usr/local/share/emailrelay/doc/reference.html
* /usr/local/share/emailrelay/doc/reference.txt
* /usr/local/share/emailrelay/doc/userguide.html
* /usr/local/share/emailrelay/doc/userguide.txt
* /usr/local/share/emailrelay/doc/windows.html
* /usr/local/share/emailrelay/doc/windows.txt
* /usr/local/var/spool/emailrelay/emailrelay.*.content
* /usr/local/var/spool/emailrelay/emailrelay.*.envelope
This directory structure is constrained by the GNU/"autoconf" conventions rather
than the Filesystem Hierarchy Standard (FHS).
To force FHS compliance you can use the "--enable-fhs" switch when running
"configure". This results in the following file locations:
* /etc/init.d/emailrelay
* /usr/lib/emailrelay/emailrelay-poke
* /usr/sbin/emailrelay
* /usr/sbin/emailrelay-passwd
* /usr/sbin/emailrelay-submit
* /usr/share/doc/emailrelay/examples/emailrelay-notify.sh
* /usr/share/doc/emailrelay/examples/emailrelay-deliver.sh
* /usr/share/doc/emailrelay/examples/emailrelay-resubmit.sh
* /usr/share/doc/emailrelay/examples/emailrelay-process.sh
* /usr/share/doc/emailrelay/developer.txt
* /usr/share/doc/emailrelay/reference.txt
* /usr/share/doc/emailrelay/userguide.txt
* /usr/share/doc/emailrelay/windows.txt
* /usr/share/doc/emailrelay/readme.html
* /usr/share/doc/emailrelay/developer.html
* /usr/share/doc/emailrelay/reference.html
* /usr/share/doc/emailrelay/userguide.html
* /usr/share/doc/emailrelay/windows.html
* /usr/share/doc/emailrelay/emailrelay-man.html
* /usr/share/doc/emailrelay/index.html
* /usr/share/doc/emailrelay/changelog.html
* /usr/share/doc/emailrelay/emailrelay.css
* /usr/share/doc/emailrelay/NEWS
* /usr/share/doc/emailrelay/README
* /usr/share/doc/emailrelay/changelog.gz
* /usr/share/doc/emailrelay/doxygen/*
* /usr/share/man/man1/emailrelay.1.gz
* /usr/share/doc/emailrelay/changelog.html
* /usr/share/doc/emailrelay/developer.html
* /usr/share/doc/emailrelay/developer.txt
* /usr/share/doc/emailrelay/emailrelay.css
* /usr/share/doc/emailrelay/examples/emailrelay-deliver.sh
* /usr/share/doc/emailrelay/examples/emailrelay-notify.sh
* /usr/share/doc/emailrelay/examples/emailrelay-process.sh
* /usr/share/doc/emailrelay/examples/emailrelay-resubmit.sh
* /usr/share/doc/emailrelay/index.html
* /usr/share/doc/emailrelay/readme.html
* /usr/share/doc/emailrelay/reference.html
* /usr/share/doc/emailrelay/reference.txt
* /usr/share/doc/emailrelay/userguide.html
* /usr/share/doc/emailrelay/userguide.txt
* /usr/share/doc/emailrelay/windows.html
* /usr/share/doc/emailrelay/windows.txt
* /usr/share/man/man1/emailrelay-passwd.1.gz
* /usr/share/man/man1/emailrelay-poke.1.gz
* /usr/share/man/man1/emailrelay-submit.1.gz
* /var/spool/emailrelay/*
* /etc/init.d/emailrelay
* /usr/share/man/man1/emailrelay.1.gz
* /var/spool/emailrelay/emailrelay.*.content
* /var/spool/emailrelay/emailrelay.*.envelope
For finer control of the directory structure the following variables
can be specified on the "configure" command line (but note that the
can be specified on the "configure" command-line (but note that the
"--enable-fhs" will override them):
* e_sbindir
* e_libexecdir
@ -451,12 +504,26 @@ and create the E-MailRelay spool directory as "/tmp/spool" rather than
"/usr/local/var/spool/emailrelay".
The default spool directory path which is built into the executables and scripts
comes from "configure", via the makefiles. So if you re-run "configure" with a
different spool directory you should also do a "make clean" in at least
"src/main" and "bin".
comes from "configure", via the makefiles. So after running "configure" with a
different spool directory do a "make clean" in at least "src/main" and "bin".
Even though the "--enable-fhs" switch overrides all other directory specifiers
during the build process, it is still possible to change the installation
root directory using "make install DESTDIR=<root>" or "DESTDIR=<root> make -e install".
However this will not affect the default spool directory path built into the
scripts and executables; the spool directory path will have to be explicitly
defined at run-time.
For example, building with "configure --enable-fhs ; make ; make install DESTDIR=/root"
would create a spool directory "/root/var/spool/emailrelay", but the server
run as "/root/usr/sbin/emailrelay --as-server" would use a expect a spool directory
of "/var/spool/emailrelay". This could be fixed by either adding
"--spool-dir /root/var/spool/emailrelay" to the server command-line, or by
creating the "/var/spool/emailrelay" directory manually.
The "emailrelay" script in "init.d" creates a pid file in "/var/run" if the
directory exists, or in "/tmp" otherwise.
Copyright (C) 2001-2002 Graeme Walker <graeme_walker@users.sourceforge.net>. All rights reserved.

View File

@ -94,7 +94,7 @@ 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.
differences between the various operating systems and distributions.
Many systems, including the most popular GNU/Linux distributions, use the
System-V mechanism for starting daemons at boot time. The directory
@ -106,6 +106,10 @@ links point back into the start/stop script in the parent directory, using a
numeric part of the link name determines the order in which the links are
called.
If you cannot use a GUI front-end as provided by GNOME and KDE, or a
command-line utility like "insserv", "chkconfig" or "/usr/lib/lsb/install_initd",
then you can follow the steps below to set up the necessary symbolic links.
Before you start you will need to know where your "init.d" directory can be
found and what your default run level is:
@ -137,8 +141,6 @@ daemons compete for the standard SMTP listening port):
$ cd /etc/init.d/rc5.d
$ rm *sendmail
(There are also KDE and GNOME GUIs which you can use to do the latter steps.)
Triggering onward delivery
--------------------------
This section assumes that you are using "pppd" to establish your dial-up
@ -160,7 +162,7 @@ If you find "sendmail -q" then it should be sufficient to replace it with this:
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:
"ip-up.local" script like this:
$ cd /etc/ppp
$ cat << EOF > ip-up.local

View File

@ -18,6 +18,7 @@ In summary, the Windows installation process is as follows:
* Create a spool directory under "<windir>\spool", eg. "c:\win98\spool\emailrelay".
* Configure your e-mail client (eg. Outlook) to use SMTP on the local machine for outgoing e-mail.
* Run the forwarding client from the taskbar/desktop shortcut once connected to the Internet.
* Optionally register the server as a COM component using the "-regserver" command-line switch.
These steps are explained in more detail below.
@ -68,29 +69,37 @@ Once the server is configured the next step is to create an icon on the desktop
you are connected to the Internet.
You can do this by simply dragging the E-MailRelay executable file from the
"Program Files\emailrelay" folder onto the desktop. Then right click on the
new icon and select "Properties" from the pop-up menu.
"Program Files\emailrelay" folder onto the desktop. Then right click on the new
icon and select "Properties" from the pop-up menu.
Again the "Target" box should contain the path of the E-MailRelay executable,
and you should add configuration options at the end, separated by a space.
As a minumum you will need to add "--as-client myisp:smtp", where "myisp"
is replaced with the hostname of your ISP's SMTP server.
and you should add configuration options at the end, separated by a space. As a
minumum you will need to add "--as-client myisp:smtp", where "myisp" is replaced
with the hostname of your ISP's SMTP server.
Preparation
-----------
To complete the installation you should create a spool directory to hold
your e-mail messages while you are off-line. By default E-MailRelay will
look for a directory "<windir>\spool\emailrelay", where "<windir>" is the
path of your main windows directory, typically "c:\win98" for Windows 98.
To complete the installation you should create a spool directory to hold your
e-mail messages while you are off-line. By default E-MailRelay will look for a
directory "<windir>\spool\emailrelay", where "<windir>" is the path of your main
windows directory, typically "c:\win98" for Windows 98.
Finally you will need to configure your e-mail client program to use
the local E-MailRelay server for outgoing mail. Where it asks for the
name of the SMTP server for outgoing mail you should tell it to use
"localhost" or "127.0.0.1".
Finally you will need to configure your e-mail client program to use the local
E-MailRelay server for outgoing mail. Where it asks for the name of the SMTP
server for outgoing mail you should tell it to use "localhost" or "127.0.0.1".
COM registration
----------------
The server can be optionally registered as a COM component by running the
executable with the "-regserver" command line switch. The COM interface may
allow stored messages to be forwarded to any remote server, so consider the
security implications before doing the COM registration. Refer to the reference
guide for more details.
Uninstall
---------
There are no DLLs or registry entries to clean up: just delete the files under
If you have ever registered the executable as a COM component, then deregister it
using the "-unregserver" command-line switch. Then just delete the files under
"Program Files\emailrelay" and "<windir>\spool\emailrelay", and remove any
taskbar, desktop or "Start->Programs->StartUp" shortcuts.

View File

@ -1,12 +1,14 @@
Summary: Simple e-mail message transfer agent using SMTP
Name: emailrelay
Version: 0.9.9
Version: 1.0.0
Release: 1
Copyright: GPL
Group: System Environment/Daemons
Source: http://emailrelay.sourceforge.net/.../emailrelay-src-0.9.9.tar.gz
Source: http://emailrelay.sourceforge.net/.../emailrelay-src-1.0.0.tar.gz
BuildRoot: /tmp/emailrelay-install
%define prefix /usr
%description
E-MailRelay is a simple SMTP store-and-forward message transfer agent (MTA).
It runs as an SMTP server, storing e-mail in a local spool directory, and
@ -17,8 +19,8 @@ than to a local postmaster. Because of this functional simplicity it is
extremely easy to configure, typically only requiring the address of the
next-hop 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.
C++ source code is available for Linux, FreeBSD (etc) and Windows.
Distribution is under the GNU General Public License.
%prep
%setup
@ -30,43 +32,49 @@ make HAVE_DOXYGEN=no HAVE_MAN2HTML=no
%install
make install destdir=$RPM_BUILD_ROOT DESTDIR=$RPM_BUILD_ROOT HAVE_DOXYGEN=no HAVE_MAN2HTML=no
%post
test -f /usr/lib/lsb/install_initd && cd /etc/init.d && /usr/lib/lsb/install_initd emailrelay
%preun
test $1 -eq 0 && test -f /usr/lib/lsb/remove_initd && cd /etc/init.d && /usr/lib/lsb/remove_initd emailrelay
%clean
rm -rf $RPM_BUILD_ROOT
%files
/usr/lib/emailrelay/emailrelay-poke
/usr/sbin/emailrelay
/usr/sbin/emailrelay-passwd
/usr/sbin/emailrelay-submit
/usr/share/doc/emailrelay/examples/emailrelay-process.sh
/usr/share/doc/emailrelay/examples/emailrelay-notify.sh
/usr/share/doc/emailrelay/examples/emailrelay-deliver.sh
/usr/share/doc/emailrelay/examples/emailrelay-resubmit.sh
/usr/share/doc/emailrelay/developer.txt
/usr/share/doc/emailrelay/reference.txt
/usr/share/doc/emailrelay/userguide.txt
/usr/share/doc/emailrelay/windows.txt
/usr/share/doc/emailrelay/readme.html
/usr/share/doc/emailrelay/developer.html
/usr/share/doc/emailrelay/reference.html
/usr/share/doc/emailrelay/userguide.html
/usr/share/doc/emailrelay/windows.html
/usr/share/doc/emailrelay/index.html
/usr/share/doc/emailrelay/changelog.html
/usr/share/doc/emailrelay/emailrelay.css
/usr/share/doc/emailrelay/NEWS
/usr/share/doc/emailrelay/README
/usr/share/doc/emailrelay/changelog.gz
/usr/share/doc/emailrelay/doxygen
/usr/share/man/man1/emailrelay.1.gz
/usr/share/man/man1/emailrelay-passwd.1.gz
/usr/share/man/man1/emailrelay-poke.1.gz
/usr/share/man/man1/emailrelay-submit.1.gz
/etc/init.d/emailrelay
%{prefix}/lib/emailrelay/emailrelay-poke
%{prefix}/sbin/emailrelay
%{prefix}/sbin/emailrelay-passwd
%{prefix}/sbin/emailrelay-submit
%{prefix}/share/doc/emailrelay/NEWS
%{prefix}/share/doc/emailrelay/README
%{prefix}/share/doc/emailrelay/changelog.gz
%{prefix}/share/doc/emailrelay/changelog.html
%{prefix}/share/doc/emailrelay/developer.html
%{prefix}/share/doc/emailrelay/developer.txt
%{prefix}/share/doc/emailrelay/emailrelay.css
%{prefix}/share/doc/emailrelay/examples/emailrelay-deliver.sh
%{prefix}/share/doc/emailrelay/examples/emailrelay-notify.sh
%{prefix}/share/doc/emailrelay/examples/emailrelay-process.sh
%{prefix}/share/doc/emailrelay/examples/emailrelay-resubmit.sh
%{prefix}/share/doc/emailrelay/index.html
%{prefix}/share/doc/emailrelay/readme.html
%{prefix}/share/doc/emailrelay/reference.html
%{prefix}/share/doc/emailrelay/reference.txt
%{prefix}/share/doc/emailrelay/userguide.html
%{prefix}/share/doc/emailrelay/userguide.txt
%{prefix}/share/doc/emailrelay/windows.html
%{prefix}/share/doc/emailrelay/windows.txt
%{prefix}/share/man/man1/emailrelay-passwd.1.gz
%{prefix}/share/man/man1/emailrelay-poke.1.gz
%{prefix}/share/man/man1/emailrelay-submit.1.gz
%{prefix}/share/man/man1/emailrelay.1.gz
/var/spool/emailrelay/
%changelog
* Mon Sep 24 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
* Wed Jul 3 2002 Graeme Walker <graeme_walker@users.sourceforge.net>
- Initial version.

View File

@ -79,6 +79,7 @@ POST_UNINSTALL = :
AR = @AR@
AWK = @AWK@
CC = @CC@
COMPILER_VERSION = @COMPILER_VERSION@
CXX = @CXX@
GZIP = @GZIP@
HAVE_DOXYGEN = @HAVE_DOXYGEN@

View File

@ -79,6 +79,7 @@ POST_UNINSTALL = :
AR = @AR@
AWK = @AWK@
CC = @CC@
COMPILER_VERSION = @COMPILER_VERSION@
CXX = @CXX@
GZIP = @GZIP@
HAVE_DOXYGEN = @HAVE_DOXYGEN@

View File

@ -20,7 +20,7 @@
//
// sstream
//
// Forwards compatibility from ostrstream to std::stringstream.
// Forwards compatibility from ostrstream to std::ostringstream.
//
#ifndef G_STD_SSTREAM_H
@ -31,30 +31,31 @@
namespace std
{
class stringstream : public ostrstream
class ostringstream : public ostrstream
{
public:
stringstream() ;
stringstream( char * , size_t ) ;
ostringstream() ;
ostringstream( char * , size_t ) ;
std::string str() const ;
} ;
typedef ostringstream stringstream ;
} ;
inline
std::stringstream::stringstream()
std::ostringstream::ostringstream()
{
}
inline
std::stringstream::stringstream( char * p , size_t n ) :
std::ostringstream::ostringstream( char * p , size_t n ) :
ostrstream(p,static_cast<int>(n))
{
}
inline
std::string std::stringstream::str() const
std::string std::ostringstream::str() const
{
std::stringstream * This = const_cast<std::stringstream*>(this) ;
std::ostringstream * This = const_cast<std::ostringstream*>(this) ;
ostrstream * base = This ;
(*base) << '\0' ;
char * p = base->str() ;

View File

@ -79,6 +79,7 @@ POST_UNINSTALL = :
AR = @AR@
AWK = @AWK@
CC = @CC@
COMPILER_VERSION = @COMPILER_VERSION@
CXX = @CXX@
GZIP = @GZIP@
HAVE_DOXYGEN = @HAVE_DOXYGEN@

View File

@ -30,6 +30,8 @@ namespace std
using ::rand ;
using ::srand ;
using ::getenv ;
using ::free ;
using ::malloc ;
} ;
#endif

View File

@ -16,4 +16,4 @@
# along with this program; if not, write to the Free Software
# Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
#
SUBDIRS = glib gnet main win32
SUBDIRS = glib gnet gsmtp main win32

View File

@ -79,6 +79,7 @@ POST_UNINSTALL = :
AR = @AR@
AWK = @AWK@
CC = @CC@
COMPILER_VERSION = @COMPILER_VERSION@
CXX = @CXX@
GZIP = @GZIP@
HAVE_DOXYGEN = @HAVE_DOXYGEN@
@ -96,7 +97,7 @@ e_man1dir = @e_man1dir@
e_sbindir = @e_sbindir@
e_spooldir = @e_spooldir@
SUBDIRS = glib gnet main win32
SUBDIRS = glib gnet gsmtp main win32
mkinstalldirs = $(SHELL) $(top_srcdir)/mkinstalldirs
CONFIG_HEADER = ../config.h
CONFIG_CLEAN_FILES =

View File

@ -16,7 +16,9 @@
# along with this program; if not, write to the Free Software
# Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
#
EXTRA_DIST=garg_win32.cpp \
EXTRA_DIST=\
garg_win32.cpp \
gcleanup_win32.cpp \
gdaemon_win32.cpp \
gdatetime_win32.cpp \
gdirectory_win32.cpp \
@ -24,9 +26,11 @@ EXTRA_DIST=garg_win32.cpp \
glogoutput_win32.cpp \
gprocess_win32.cpp \
gfile_win32.cpp \
gregistry_win32.cpp \
gregistry.h \
md5c.c \
md5.h
INCLUDES = -I$(top_srcdir)/lib/gcc2.95
INCLUDES = -I$(top_srcdir)/lib/$(COMPILER_VERSION)
noinst_LIBRARIES = libglib.a
libglib_a_SOURCES = \
garg.cpp \
@ -34,9 +38,9 @@ libglib_a_SOURCES = \
garg_unix.cpp \
gassert.h \
gconvert.h \
gcredentials.h \
gcleanup.h \
gcleanup_unix.cpp \
gdaemon.h \
gdaemon.cpp \
gdaemon_unix.cpp \
gdate.cpp \
gdate.h \
@ -68,10 +72,13 @@ libglib_a_SOURCES = \
gnoncopyable.h \
gpath.cpp \
gpath.h \
gpidfile.cpp \
gpidfile.h \
gprocess.h \
gprocess_unix.cpp \
groot.cpp \
groot.h \
gsetter.h \
gstatemachine.h \
gstr.cpp \
gstr.h \

View File

@ -79,6 +79,7 @@ POST_UNINSTALL = :
AR = @AR@
AWK = @AWK@
CC = @CC@
COMPILER_VERSION = @COMPILER_VERSION@
CXX = @CXX@
GZIP = @GZIP@
HAVE_DOXYGEN = @HAVE_DOXYGEN@
@ -96,11 +97,11 @@ e_man1dir = @e_man1dir@
e_sbindir = @e_sbindir@
e_spooldir = @e_spooldir@
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 md5c.c md5.h
EXTRA_DIST = garg_win32.cpp gcleanup_win32.cpp gdaemon_win32.cpp gdatetime_win32.cpp gdirectory_win32.cpp gfs_win32.cpp glogoutput_win32.cpp gprocess_win32.cpp gfile_win32.cpp gregistry_win32.cpp gregistry.h md5c.c md5.h
INCLUDES = -I$(top_srcdir)/lib/gcc2.95
INCLUDES = -I$(top_srcdir)/lib/$(COMPILER_VERSION)
noinst_LIBRARIES = libglib.a
libglib_a_SOURCES = garg.cpp garg.h garg_unix.cpp gassert.h gconvert.h gcredentials.h gdaemon.h gdaemon.cpp 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 gmd5.cpp gmd5.h gmemory.h gnoncopyable.h gpath.cpp gpath.h gprocess.h gprocess_unix.cpp groot.cpp groot.h gstatemachine.h gstr.cpp gstr.h gstrings.h gtime.cpp gtime.h
libglib_a_SOURCES = garg.cpp garg.h garg_unix.cpp gassert.h gconvert.h gcleanup.h gcleanup_unix.cpp 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 gmd5.cpp gmd5.h gmemory.h gnoncopyable.h gpath.cpp gpath.h gpidfile.cpp gpidfile.h gprocess.h gprocess_unix.cpp groot.cpp groot.h gsetter.h gstatemachine.h gstr.cpp gstr.h gstrings.h gtime.cpp gtime.h
mkinstalldirs = $(SHELL) $(top_srcdir)/mkinstalldirs
CONFIG_HEADER = ../../config.h
@ -113,11 +114,11 @@ CPPFLAGS = @CPPFLAGS@
LDFLAGS = @LDFLAGS@
LIBS = @LIBS@
libglib_a_LIBADD =
libglib_a_OBJECTS = garg.o garg_unix.o gdaemon.o gdaemon_unix.o gdate.o \
gdatetime.o gdatetime_unix.o gdirectory.o gdirectory_unix.o \
libglib_a_OBJECTS = garg.o garg_unix.o gcleanup_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 gmd5.o gpath.o gprocess_unix.o groot.o \
gstr.o gtime.o
glogoutput.o glogoutput_unix.o gmd5.o gpath.o gpidfile.o \
gprocess_unix.o groot.o gstr.o gtime.o
CXXFLAGS = @CXXFLAGS@
CXXCOMPILE = $(CXX) $(DEFS) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS)
CXXLD = $(CXX)
@ -227,97 +228,93 @@ distdir: $(DISTFILES)
fi; \
done
garg.o: garg.cpp gdef.h ../../config.h ../../lib/gcc2.95/iostream \
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
../../lib/gcc2.95/limits garg.h gpath.h gstrings.h gstr.h \
gexception.h gdebug.h glogoutput.h glog.h gassert.h
../../lib/gcc2.95/sstream ../../lib/gcc2.95/limits garg.h \
gpath.h gstrings.h gstr.h gexception.h gdebug.h glogoutput.h \
glog.h gassert.h
garg_unix.o: garg_unix.cpp gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits garg.h \
gdebug.h glogoutput.h glog.h gassert.h
gdaemon.o: gdaemon.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 groot.h gprocess.h gnoncopyable.h
../../lib/gcc2.95/limits garg.h gdebug.h glogoutput.h glog.h \
gassert.h
gcleanup_unix.o: gcleanup_unix.cpp gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/limits gcleanup.h gpath.h gstrings.h \
gexception.h gprocess.h groot.h gnoncopyable.h glog.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 gprocess.h
../../lib/gcc2.95/limits gdaemon.h gexception.h gpidfile.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 \
gdebug.h glogoutput.h glog.h gassert.h
../../lib/gcc2.95/sstream ../../lib/gcc2.95/limits gdate.h \
gdatetime.h gexception.h gdebug.h glogoutput.h glog.h gassert.h
gdatetime.o: gdatetime.cpp gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gdatetime.h \
gexception.h gstr.h gstrings.h gassert.h glogoutput.h glog.h
../../lib/gcc2.95/limits gdatetime.h gexception.h gstr.h \
gstrings.h gassert.h glogoutput.h glog.h
gdatetime_unix.o: gdatetime_unix.cpp gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gdatetime.h \
gexception.h
../../lib/gcc2.95/limits gdatetime.h gexception.h
gdirectory.o: gdirectory.cpp gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gdirectory.h \
gpath.h gstrings.h gexception.h gfs.h glog.h
../../lib/gcc2.95/limits gdirectory.h gpath.h gstrings.h \
gexception.h gfs.h glog.h
gdirectory_unix.o: gdirectory_unix.cpp gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gdirectory.h \
gpath.h gstrings.h gexception.h gfs.h gfile.h gdebug.h \
glogoutput.h glog.h gassert.h
../../lib/gcc2.95/limits gdirectory.h gpath.h gstrings.h \
gexception.h gfs.h gfile.h gdebug.h glogoutput.h glog.h \
gassert.h
gexception.o: gexception.cpp gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gexception.h
../../lib/gcc2.95/limits gexception.h
gfile.o: gfile.cpp gdef.h ../../config.h ../../lib/gcc2.95/iostream \
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
../../lib/gcc2.95/limits gfile.h gpath.h gstrings.h \
gexception.h gprocess.h glog.h
../../lib/gcc2.95/sstream ../../lib/gcc2.95/limits gfile.h \
gpath.h gstrings.h gexception.h gprocess.h glog.h
gfile_unix.o: gfile_unix.cpp gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gfile.h \
gpath.h gstrings.h gexception.h gprocess.h
../../lib/gcc2.95/limits gfile.h gpath.h gstrings.h \
gexception.h gprocess.h
gfs_unix.o: gfs_unix.cpp gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gfs.h
../../lib/gcc2.95/limits gfs.h
ggetopt.o: ggetopt.cpp gdef.h ../../config.h ../../lib/gcc2.95/iostream \
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
../../lib/gcc2.95/limits glog.h gstrings.h gstr.h gexception.h \
ggetopt.h garg.h gassert.h glogoutput.h gdebug.h
../../lib/gcc2.95/sstream ../../lib/gcc2.95/limits glog.h \
gstrings.h gstr.h gexception.h ggetopt.h garg.h gassert.h \
glogoutput.h gdebug.h
glog.o: glog.cpp gdef.h ../../config.h ../../lib/gcc2.95/iostream \
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
../../lib/gcc2.95/limits glog.h glogoutput.h
../../lib/gcc2.95/sstream ../../lib/gcc2.95/limits glog.h \
glogoutput.h
glogoutput.o: glogoutput.cpp gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits glogoutput.h \
glog.h
../../lib/gcc2.95/limits glogoutput.h glog.h
glogoutput_unix.o: glogoutput_unix.cpp gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits glogoutput.h \
glog.h
../../lib/gcc2.95/limits glogoutput.h glog.h
gmd5.o: gmd5.cpp gdef.h ../../config.h ../../lib/gcc2.95/iostream \
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
../../lib/gcc2.95/limits gmd5.h gexception.h gstr.h gstrings.h \
gassert.h glogoutput.h glog.h md5.h md5c.c
../../lib/gcc2.95/sstream ../../lib/gcc2.95/limits gmd5.h \
gexception.h gstr.h gstrings.h gassert.h glogoutput.h glog.h \
md5.h md5c.c
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
../../lib/gcc2.95/sstream ../../lib/gcc2.95/limits gpath.h \
gstrings.h gfs.h gstr.h gexception.h gdebug.h glogoutput.h \
glog.h gassert.h
gpidfile.o: gpidfile.cpp gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/limits gpidfile.h gexception.h gpath.h \
gstrings.h gprocess.h gcleanup.h gfile.h gdebug.h glogoutput.h \
glog.h gassert.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 gprocess.h \
gexception.h gpath.h gstrings.h gassert.h glogoutput.h glog.h \
gfs.h
../../lib/gcc2.95/limits gprocess.h gexception.h gpath.h \
gstrings.h gassert.h glogoutput.h glog.h gfs.h
groot.o: groot.cpp gdef.h ../../config.h ../../lib/gcc2.95/iostream \
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
../../lib/gcc2.95/limits groot.h gprocess.h gexception.h \
gpath.h gstrings.h gnoncopyable.h gdebug.h glogoutput.h glog.h \
gassert.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 \
../../lib/gcc2.95/sstream ../../lib/gcc2.95/limits groot.h \
gprocess.h gexception.h gpath.h gstrings.h gnoncopyable.h \
gdebug.h glogoutput.h glog.h gassert.h
gstr.o: gstr.cpp gdef.h ../../config.h ../../lib/gcc2.95/iostream \
../../lib/gcc2.95/sstream ../../lib/gcc2.95/limits gstr.h \
gexception.h gstrings.h gdebug.h glogoutput.h glog.h gassert.h
gtime.o: gtime.cpp gdef.h ../../config.h ../../lib/gcc2.95/iostream \
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
../../lib/gcc2.95/limits gtime.h gexception.h gdatetime.h \
gassert.h glogoutput.h glog.h
../../lib/gcc2.95/sstream ../../lib/gcc2.95/limits gtime.h \
gexception.h gdatetime.h gassert.h glogoutput.h glog.h
info-am:
info: info-am

View File

@ -77,16 +77,24 @@ void G::Arg::parse( HINSTANCE hinstance , const std::string & command_line )
setPrefix() ;
}
bool G::Arg::contains( const std::string & sw , size_t sw_args ) const
void G::Arg::reparse( const std::string & command_line )
{
return find( sw , sw_args , NULL ) ;
while( m_array.size() > 1U )
m_array.pop_back() ;
G::Str::splitIntoTokens( command_line , m_array , " \t" ) ;
}
bool G::Arg::find( const std::string & sw , size_t sw_args , size_t * index_p ) const
bool G::Arg::contains( const std::string & sw , size_t sw_args , bool cs ) const
{
return find( cs , sw , sw_args , NULL ) ;
}
bool G::Arg::find( bool cs , const std::string & sw , size_t sw_args ,
size_t * index_p ) const
{
for( size_t i = 1 ; i < m_array.size() ; i++ ) // start from v[1]
{
if( sw == m_array[i] && (i+sw_args) < m_array.size() )
if( match(cs,sw,m_array[i]) && (i+sw_args) < m_array.size() )
{
if( index_p != NULL )
*index_p = i ;
@ -96,10 +104,19 @@ bool G::Arg::find( const std::string & sw , size_t sw_args , size_t * index_p )
return false ;
}
//static
bool G::Arg::match( bool cs , const std::string & s1 , const std::string & s2 )
{
return
cs ?
s1 == s2 :
Str::upper(s1) == Str::upper(s2) ;
}
void G::Arg::remove( const std::string & sw , size_t sw_args )
{
size_t i = 0U ;
const bool found = find( sw , sw_args , &i ) ;
const bool found = find( true , sw , sw_args , &i ) ;
if( found )
removeAt( i , sw_args ) ;
}
@ -119,7 +136,7 @@ void G::Arg::removeAt( size_t sw_index , size_t sw_args )
size_t G::Arg::index( const std::string & sw , size_t sw_args ) const
{
size_t i = 0U ;
const bool found = find( sw , sw_args , &i ) ;
const bool found = find( true , sw , sw_args , &i ) ;
return found ? i : 0U ;
}

View File

@ -31,7 +31,7 @@
namespace G
{
class Arg ;
} ;
}
// Class: G::Arg
// Description: A class which holds a represention of the
@ -57,8 +57,14 @@ public:
//
// Parses the given command line, splitting
// it up into an array of tokens.
// The program name is automatically
// The full exe name is automatically
// added as the first token (cf. argv[0]).
void reparse( const std::string & command_line ) ;
// Reinitialises the object with the given
// command-line. The command-line should not
// contain the program name: the v(0) value
// and prefix() are unchanged.
~Arg() ;
// Destructor.
@ -81,11 +87,12 @@ public:
// be used in main() outside of the outermost try
// block.
bool contains( const std::string & sw , size_t sw_args = 0U ) const ;
// Returns true if the command line
// contains the given switch with enough
// command line arguments left to satisfy
// the given number of switch arguments.
bool contains( const std::string & sw ,
size_t sw_args = 0U , bool case_sensitive = true ) const ;
// Returns true if the command line
// contains the given switch with enough
// command line arguments left to satisfy
// the given number of switch arguments.
size_t index( const std::string & sw , size_t sw_args = 0U ) const ;
// Returns the index of the given switch.
@ -109,8 +116,9 @@ public:
private:
static std::string moduleName( HINSTANCE h ) ;
bool find( const std::string & sw , size_t sw_args , size_t *index_p ) const ;
bool find( bool , const std::string & , size_t , size_t * ) const ;
void setPrefix() ;
static bool match( bool , const std::string & , const std::string & ) ;
private:
typedef std::vector<std::string GAllocator(std::string) > Array ;

57
src/glib/gcleanup.h Normal file
View File

@ -0,0 +1,57 @@
//
// Copyright (C) 2001-2002 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.
//
// ===
//
// gcleanup.h
//
#ifndef G_CLEANUP_H
#define G_CLEANUP_H
#include "gdef.h"
#include "gpath.h"
#include "gexception.h"
namespace G
{
class Cleanup ;
}
// Class: G::Cleanup
// Description: An interface for registering cleanup functions
// which are called when the process terminates abnormally.
//
class G::Cleanup
{
public:
G_EXCEPTION( Error , "cleanup error" ) ;
static void add( void (*fn)(const char*) , const char * arg ) ;
// Adds the given handler to the list which
// are to be called when the process
// terminates abnormally. The handler
// function must be fully reentrant.
// The 'arg' pointer is kept.
private:
Cleanup() ; // not implemeneted
} ;
#endif

154
src/glib/gcleanup_unix.cpp Normal file
View File

@ -0,0 +1,154 @@
//
// Copyright (C) 2001-2002 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.
//
// ===
//
// gcleanup_unix.cpp
//
#include "gdef.h"
#include "gcleanup.h"
#include "gprocess.h"
#include "groot.h"
#include "glog.h"
#include <signal.h>
namespace G
{
class CleanupImp ;
}
// Class: G::CleanupImp
// Description: A private implementation class used by G::Cleanup.
//
class G::CleanupImp
{
public:
static void add( void (*fn)(const char*) , const char * ) ;
private:
static void init() ;
static void install( int , void (*fn)(int) ) ;
static void installHandler( int ) ;
static void installDefault( int ) ;
static void callHandlers() ;
static void handler_( int ) ;
static bool ignored( int ) ;
private:
typedef Cleanup::Error Error ;
struct Link // A private linked-list structure used by G::CleanupImp.
{
void (*fn)(const char*) ;
const char * arg ;
Link * next ;
} ;
static Link * m_head ;
static Link * m_tail ;
} ;
G::CleanupImp::Link * G::CleanupImp::m_head = NULL ;
G::CleanupImp::Link * G::CleanupImp::m_tail = NULL ;
// ===
void G::Cleanup::add( void (*fn)(const char*) , const char * arg )
{
CleanupImp::add( fn , arg ) ;
}
// ===
void G::CleanupImp::init()
{
installHandler( SIGTERM ) ;
installHandler( SIGINT ) ;
installHandler( SIGHUP ) ;
installHandler( SIGPIPE ) ;
installHandler( SIGQUIT ) ;
//installHandler( SIGUSR1 ) ;
//installHandler( SIGUSR2 ) ;
}
void G::CleanupImp::add( void (*fn)(const char*) , const char * arg )
{
Link * p = new Link ;
p->fn = fn ;
p->arg = arg ;
p->next = NULL ;
// (this bit should be protected from signals)
if( m_head == NULL ) init() ;
if( m_tail != NULL ) m_tail->next = p ;
m_tail = p ;
if( m_head == NULL ) m_head = p ;
}
void G::CleanupImp::installHandler( int signum )
{
if( ignored(signum) )
G_DEBUG( "G::CleanupImp::installHandler: signal " << signum << " is ignored" ) ;
else
install( signum , handler_ ) ;
}
void G::CleanupImp::installDefault( int signum )
{
install( signum , NULL ) ;
}
bool G::CleanupImp::ignored( int signum )
{
static struct ::sigaction zero_action ;
struct ::sigaction action( zero_action ) ;
if( ::sigaction( signum , NULL , &action ) )
throw Error( "sigaction" ) ;
return action.sa_handler == SIG_IGN ;
}
void G::CleanupImp::install( int signum , void (*fn)(int) )
{
static struct ::sigaction zero_action ;
struct ::sigaction action( zero_action ) ;
action.sa_handler = fn ? fn : SIG_DFL ;
if( ::sigaction( signum , &action , NULL ) && fn != NULL )
throw Error( "sigaction" ) ;
}
void G::CleanupImp::callHandlers()
{
G::Root claim_root ;
for( const Link * p = m_head ; p != NULL ; p = p->next )
{
(*(p->fn))(p->arg) ;
}
}
void G::CleanupImp::handler_( int signum )
{
try
{
callHandlers() ;
installDefault( signum ) ;
::raise( signum ) ;
}
catch(...)
{
}
}

View File

@ -18,16 +18,14 @@
//
// ===
//
// gevent_unix.cpp
// gcleanup_win32.cpp
//
#include "gdef.h"
#include "gnet.h"
#include "geventloop.h"
#include "gselect.h"
#include "gcleanup.h"
GNet::EventLoop * GNet::EventLoop::create()
void G::Cleanup::add( void (*)(const char*) , const char * )
{
return new Select ;
// not implemented
}

View File

@ -44,11 +44,11 @@ Tout Convert( const Tin & in )
Tout out = in ;
Tin copy = out ;
if( in != copy )
throw ConvertOverflow( std::stringstream() << in ) ;
throw ConvertOverflow( std::ostringstream() << in ) ;
return out ;
}
} ;
}
#endif

View File

@ -1,79 +0,0 @@
//
// Copyright (C) 2001-2002 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.
//
// ===
//
// gcredentials.h
//
#ifndef G_CREDENTIALS_H
#define G_CREDENTIALS_H
#include "gdef.h"
namespace G
{
// Class: credentials
// Description: A class template which can be used to provide
// controlled access to public methods in a class. Note that
// all the constructors are private, but the template-parameter
// class is declared as a friend. (Because all the methods
// are private you will not see much in the doxygen output;
// have a look at the header.)
//
// Usage:
/// struct Foo
/// {
/// void methodForBar( const credentials<Bar> & ) ;
/// } ;
///
/// void Bar::fn()
/// {
/// Foo foo ;
/// foo.methodForBar( "" ) ;
/// }
//
template <class T>
class credentials
{
friend T ;
credentials() ;
// Private default constructor.
credentials( const char * ) ;
// Implicit private constructor.
} ;
template <class T>
inline
credentials<T>::credentials()
{
}
template <class T>
inline
credentials<T>::credentials( const char * )
{
}
} ;
#endif

View File

@ -1,75 +0,0 @@
//
// Copyright (C) 2001-2002 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.
//
// ===
//
// gdaemon.cpp
//
#include "gdef.h"
#include "gdaemon.h"
#include "groot.h"
#include "gprocess.h"
//static
void G::Daemon::PidFile::check( 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()) ;
}
//static
void G::Daemon::PidFile::test( 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()) ;
}
}
//static
void G::Daemon::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()) ;
}
}
G::Daemon::PidFile::PidFile() :
m_valid(false)
{
}
G::Daemon::PidFile::PidFile( const G::Path & path ) :
m_path(path) ,
m_valid(true)
{
}
void G::Daemon::PidFile::commit()
{
if( m_valid )
create( m_path ) ;
}

View File

@ -26,6 +26,7 @@
#include "gdef.h"
#include "gexception.h"
#include "gpidfile.h"
#include "gpath.h"
#include <sys/types.h>
#include <string>
@ -33,7 +34,7 @@
namespace G
{
class Daemon ;
} ;
}
// Class: G::Daemon
// Description: A class for deamonising the calling process.
@ -46,19 +47,6 @@ class G::Daemon
{
public:
G_EXCEPTION( CannotFork , "cannot fork" ) ;
G_EXCEPTION( BadPidFile , "invalid pid file" ) ;
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 ;
private: static void check( const Path & ) ;
private: static void test( const Path & ) ;
private: static void create( const Path & ) ;
} ;
static void detach() ;
// Detaches from the parent environment.
@ -66,24 +54,19 @@ public:
// _exit()ing the parent, and calling
// setsid() in the child.
static void detach( const Path & pid_file ) ;
// An overload which writes the new process-id
// to a file. The path must be absolute.
// Throws BadPidFile on error.
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.
// of the new process-id to a file.
//
// 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.
// which open a listening port. A second instance
// of the process will fail on startup, and should
// not overwrite the pid file of the running
// server. In this situation PidFile::commit()
// should be called just before entering the event
// loop.
//
// Throws BadPidFile on error.
// Throws PidFile::Error on error.
private:
Daemon() ;

View File

@ -25,21 +25,10 @@
#include "gdaemon.h"
#include "gprocess.h"
//static
void G::Daemon::detach( const Path & pid_file )
{
PidFile::check( pid_file ) ;
PidFile::test( pid_file ) ;
detach() ;
PidFile::create( pid_file ) ;
}
//static
void G::Daemon::detach( PidFile & pid_file )
{
PidFile::check( pid_file.m_path ) ;
pid_file.check() ; // absolute path
detach() ;
}

View File

@ -25,12 +25,6 @@
#include "gdaemon.h"
#include "gprocess.h"
//static
void G::Daemon::detach( const Path & )
{
detach() ;
}
//static
void G::Daemon::detach( PidFile & )
{

View File

@ -86,7 +86,7 @@ void G::Date::init( const G::DateTime::BrokenDownTime & tm )
std::string G::Date::string( Format format ) const
{
std::stringstream ss ;
std::ostringstream ss ;
if( format == yyyy_mm_dd_slash )
ss << m_year << "/" << m_month << "/" << m_day ;
else
@ -104,7 +104,7 @@ int G::Date::monthday() const
std::string G::Date::monthdayString() const
{
std::stringstream ss ;
std::ostringstream ss ;
ss << m_day ;
return ss.str() ;
}
@ -173,7 +173,7 @@ int G::Date::year() const
std::string G::Date::yearString() const
{
std::stringstream ss ;
std::ostringstream ss ;
ss << m_year ;
return ss.str() ;
}

View File

@ -34,7 +34,7 @@
namespace G
{
class Date ;
} ;
}
// Class: G::Date
// Description: A date (dd/mm/yyyy) class.

View File

@ -81,7 +81,7 @@ std::string G::DateTime::offsetString( Offset offset )
unsigned int hh = offset.second / 3600U ;
unsigned int mm = (offset.second / 60U) % 60 ;
std::stringstream ss ;
std::ostringstream ss ;
char sign = (offset.first || (hh==0&&mm==0)) ? '+' : '-' ;
ss << sign << (hh/10U) << (hh%10U) << (mm/10) << (mm%10) ;
return ss.str() ;

View File

@ -32,7 +32,7 @@
namespace G
{
class DateTime ;
} ;
}
// Class: G::DateTime
// Description: A low-level static class used by Date and Time.

View File

@ -26,11 +26,11 @@
std::tm * G::DateTime::gmtime_r( const std::time_t * t , std::tm * p )
{
return std::gmtime_r(t,p) ;
return ::gmtime_r(t,p) ;
}
std::tm * G::DateTime::localtime_r( const std::time_t * t , std::tm * p )
{
return std::localtime_r(t,p) ;
return ::localtime_r(t,p) ;
}

View File

@ -40,26 +40,24 @@
#if ! HAVE_GMTIME_R
#include <ctime>
namespace std
inline struct tm * gmtime_r( const time_t * tp , struct tm * tm_p )
{
inline struct tm * gmtime_r( const time_t * tp , struct tm * tm_p )
{
* tm_p = * gmtime( tp ) ;
return tm_p ;
}
} ;
* tm_p = * gmtime( tp ) ;
return tm_p ;
}
#endif
#if ! HAVE_LOCALTIME_R
#include <ctime>
namespace std
inline struct tm * localtime_r( const time_t * tp , struct tm * tm_p )
{
inline struct tm * localtime_r( const time_t * tp , struct tm * tm_p )
{
* tm_p = * localtime( tp ) ;
return tm_p ;
}
} ;
* tm_p = * localtime( tp ) ;
return tm_p ;
}
#endif
#if ! HAVE_SOCKLEN_T
typedef int socklen_t ;
#endif
#if ! defined( G_UNIX )
@ -119,7 +117,7 @@
#include <fstream>
#include <sstream>
#include <string>
#include <xlocale>
//#include <xlocale>
#include <limits>
#include <memory>
#include <exception>
@ -141,6 +139,7 @@
typedef int ssize_t ;
typedef int uid_t ;
typedef int gid_t ;
typedef unsigned int pid_t ;
#endif
// STL portability macros (no longer necessary)
@ -155,7 +154,7 @@
// Modify compiler error handling
//
#if G_COMPILER_IS_MICROSOFT
#ifdef G_COMPILER_IS_MICROSOFT
#pragma warning( disable : 4100 ) // unused formal parameter
#pragma warning( disable : 4355 ) // 'this' in initialiser list
#pragma warning( disable : 4511 ) // cannot create default copy ctor

View File

@ -35,7 +35,7 @@ namespace G
class DirectoryIteratorImp ;
class Directory ;
class DirectoryIterator ;
} ;
}
// Class: G::Directory
// Description: An encapsulation of a file system directory

View File

@ -34,7 +34,7 @@
namespace G
{
class DirectoryIteratorImp ;
} ;
}
bool G::Directory::valid( bool for_creation ) const
{
@ -96,7 +96,7 @@ public:
private:
void operator=( const DirectoryIteratorImp & ) ;
DirectoryIteratorImp( const DirectoryIteratorImp & ) ;
static int onError( const char * path , int errno ) ;
static int onError( const char * path , int errno_ ) ;
} ;
// ===
@ -172,14 +172,13 @@ G::DirectoryIteratorImp::DirectoryIteratorImp( const Directory &dir ,
int flags = 0 | GLOB_ERR ;
int error = ::glob( wild_path.pathCstr() , flags , onError , &m_glob ) ;
if( error )
if( error || m_glob.gl_pathv == NULL )
{
G_DEBUG( "G::DirectoryIteratorImp::ctor: glob() error: " << error ) ;
m_error = true ;
}
else
{
G_ASSERT( m_glob.gl_pathv != NULL ) ;
G_ASSERT( m_glob.gl_pathc == 0 || m_glob.gl_pathv[0] != NULL ) ;
}
}

View File

@ -281,7 +281,7 @@ std::string G::DirectoryIteratorImp::modificationTimeString() const
const DWORD & hi = m_context.ftLastWriteTime.dwHighDateTime ;
const DWORD & lo = m_context.ftLastWriteTime.dwLowDateTime ;
std::stringstream ss ;
std::ostringstream ss ;
ss << hi << std::setw(8) << std::setfill('0') << lo ;
return ss.str() ;
}

View File

@ -38,11 +38,11 @@ G::Exception::Exception( const std::string & what ) :
{
}
G::Exception::~Exception()
G::Exception::~Exception() throw()
{
}
const char * G::Exception::what() const
const char * G::Exception::what() const throw()
{
return m_what.c_str() ;
}

View File

@ -31,7 +31,7 @@
namespace G
{
class Exception ;
} ;
}
// Class: G::Exception
// Description: A general-purpose exception class derived from std::exception
@ -52,10 +52,10 @@ public:
explicit Exception( const std::string & what ) ;
// Constructor.
virtual ~Exception() ;
virtual ~Exception() throw() ;
// Destructor.
virtual const char * what() const ;
virtual const char * what() const throw() ;
// Override from std::exception.
void prepend( const char * context ) ;
@ -79,7 +79,7 @@ public:
// This method allows a derived-class exception
// to be constructed and thrown on one
// line using iostream formatting.
// Eg. throw Error( std::stringstream() << a << b ) ;
// Eg. throw Error( std::ostringstream() << a << b ) ;
} ;
#define G_EXCEPTION( class_name , description ) class class_name : public G::Exception { public: class_name() { m_what = description ; } public: explicit class_name ( std::ostream & stream ) { m_what = description ; append(stream) ; } public: explicit class_name( const char * more ) { m_what = description ; append(more) ; } public: explicit class_name( const std::string & more ) { m_what = description ; append(more) ; } }

View File

@ -33,7 +33,7 @@ namespace G
{
class File ;
class DirectoryIteratorImp ;
} ;
}
// Class: G::File
// Description: A simple static class for dealing with files.

View File

@ -54,7 +54,7 @@ std::string G::File::sizeString( const Path & path )
if( 0 != ::stat( path.pathCstr() , &statbuf ) )
return std::string() ;
std::stringstream ss ;
std::ostringstream ss ;
ss << statbuf.st_size ;
return ss.str() ;
}

View File

@ -29,7 +29,7 @@
namespace G
{
class FileSystem ;
} ;
}
// Class: G::FileSystem
// Description: Provides information about the

View File

@ -28,6 +28,7 @@
#include "ggetopt.h"
#include "gassert.h"
#include "gdebug.h"
#include <sstream>
G::GetOpt::GetOpt( const Arg & args_in , const std::string & spec ,
char sep_major , char sep_minor , char escape ) :
@ -46,18 +47,25 @@ void G::GetOpt::parseSpec( const std::string & spec , char sep_major , char sep_
for( Strings::iterator p = outer.begin() ; p != outer.end() ; ++p )
{
if( (*p).empty() ) continue ;
StringArray inner ;
std::string ws_minor( 1U , sep_minor ) ;
G::Str::splitIntoFields( *p , inner , ws_minor , escape ) ;
if( inner.size() != 5U )
throw InvalidSpecification(std::stringstream() << "\"" << *p << "\" (" << ws_minor << ")") ;
if( inner.size() != 6U )
{
std::ostringstream ss ;
ss << "\"" << *p << "\" (" << ws_minor << ")" ;
throw InvalidSpecification( ss.str() ) ;
}
bool valued = G::Str::toUInt( inner[3U] ) != 0U ;
addSpec( inner[1U] , inner[0U].at(0U) , inner[1U] , inner[2U] , valued , inner[4U] ) ;
unsigned int level = G::Str::toUInt( inner[5U] ) ;
addSpec( inner[1U] , inner[0U].at(0U) , inner[1U] , inner[2U] , valued , inner[4U] , level ) ;
}
}
void G::GetOpt::addSpec( const std::string & sort_key , char c , const std::string & name ,
const std::string & description , bool valued , const std::string & value_description )
const std::string & description , bool valued , const std::string & value_description ,
unsigned int level )
{
if( c == '\0' )
throw InvalidSpecification() ;
@ -65,17 +73,22 @@ void G::GetOpt::addSpec( const std::string & sort_key , char c , const std::stri
const bool debug = true ;
if( debug )
{
G_DEBUG( "G::GetOpt::addSpec: "
std::ostringstream ss ;
ss
<< "G::GetOpt::addSpec: "
<< "sort-key=" << sort_key << ": "
<< "char=" << c << ": "
<< "name=" << name << ": "
<< "description=" << description << ": "
<< "valued=" << (valued?"true":"false") << ": "
<< "value-description=" << value_description ) ;
<< "value-description=" << value_description << ": "
<< "level=" << level ;
G_DEBUG( ss.str() ) ;
}
std::pair<SwitchSpecMap::iterator,bool> rc =
m_spec_map.insert( std::make_pair(sort_key,SwitchSpec(c,name,description,valued,value_description)) ) ;
m_spec_map.insert( std::make_pair( sort_key ,
SwitchSpec(c,name,description,valued,value_description,level) ) ) ;
if( ! rc.second )
throw InvalidSpecification("duplication") ;
@ -115,28 +128,41 @@ size_t G::GetOpt::wrapDefault()
return 79U ;
}
//static
size_t G::GetOpt::tabDefault()
{
return 30U ;
}
//static
G::GetOpt::Level G::GetOpt::levelDefault()
{
return Level(99U) ;
}
//static
size_t G::GetOpt::widthLimit( size_t w )
{
return (w != 0U && w < 50U) ? 50U : w ;
}
void G::GetOpt::showUsage( std::ostream & stream , const std::string & args ) const
void G::GetOpt::showUsage( std::ostream & stream , const std::string & args , bool verbose ) const
{
showUsage( stream , m_args.prefix() , args ) ;
showUsage( stream , m_args.prefix() , args , verbose ? levelDefault() : Level(1U) ) ;
}
void G::GetOpt::showUsage( std::ostream & stream , const std::string & exe , const std::string & args ,
size_t tab_stop , size_t width ) const
Level level , size_t tab_stop , size_t width ) const
{
stream
<< usageSummary(exe,args,width) << std::endl
<< usageHelp(tab_stop,width) ;
<< usageSummary(exe,args,level,width) << std::endl
<< usageHelp(level,tab_stop,width,false) ;
}
std::string G::GetOpt::usageSummary( const std::string & exe , const std::string & args , size_t width ) const
std::string G::GetOpt::usageSummary( const std::string & exe , const std::string & args ,
Level level , size_t width ) const
{
std::string s = std::string("usage: ") + exe + " " + usageSummarySwitches() + args ;
std::string s = std::string("usage: ") + exe + " " + usageSummarySwitches(level) + args ;
if( width != 0U )
{
return G::Str::wrap( s , "" , " " , widthLimit(width) ) ;
@ -147,19 +173,28 @@ std::string G::GetOpt::usageSummary( const std::string & exe , const std::string
}
}
std::string G::GetOpt::usageSummarySwitches() const
std::string G::GetOpt::usageSummarySwitches( Level level ) const
{
return usageSummaryPartOne() + usageSummaryPartTwo() ;
return usageSummaryPartOne(level) + usageSummaryPartTwo(level) ;
}
std::string G::GetOpt::usageSummaryPartOne() const
//static
bool G::GetOpt::visible( SwitchSpecMap::const_iterator p , Level level , bool exact )
{
return
exact ?
( !(*p).second.hidden && (*p).second.level == level.level ) :
( !(*p).second.hidden && (*p).second.level <= level.level ) ;
}
std::string G::GetOpt::usageSummaryPartOne( Level level ) const
{
// summarise the single-character switches, excluding those which take a value
std::stringstream ss ;
std::ostringstream ss ;
bool first = true ;
for( SwitchSpecMap::const_iterator p = m_spec_map.begin() ; p != m_spec_map.end() ; ++p )
{
if( !(*p).second.valued && !(*p).second.hidden )
if( !(*p).second.valued && visible(p,level,false) )
{
if( first )
ss << "[-" ;
@ -173,13 +208,13 @@ std::string G::GetOpt::usageSummaryPartOne() const
return s ;
}
std::string G::GetOpt::usageSummaryPartTwo() const
std::string G::GetOpt::usageSummaryPartTwo( Level level ) const
{
std::stringstream ss ;
std::ostringstream ss ;
const char * sep = "" ;
for( SwitchSpecMap::const_iterator p = m_spec_map.begin() ; p != m_spec_map.end() ; ++p )
{
if( !(*p).second.hidden )
if( visible(p,level,false) )
{
ss << sep << "[" ;
if( (*p).second.name.length() )
@ -203,17 +238,18 @@ std::string G::GetOpt::usageSummaryPartTwo() const
return ss.str() ;
}
std::string G::GetOpt::usageHelp( size_t tab_stop , size_t width ) const
std::string G::GetOpt::usageHelp( Level level , size_t tab_stop , size_t width , bool exact ) const
{
return usageHelpCore( " " , tab_stop , widthLimit(width) ) ;
return usageHelpCore( " " , level , tab_stop , widthLimit(width) , exact ) ;
}
std::string G::GetOpt::usageHelpCore( const std::string & prefix , size_t tab_stop , size_t width ) const
std::string G::GetOpt::usageHelpCore( const std::string & prefix , Level level ,
size_t tab_stop , size_t width , bool exact ) const
{
std::string result ;
for( SwitchSpecMap::const_iterator p = m_spec_map.begin() ; p != m_spec_map.end() ; ++p )
{
if( !(*p).second.hidden )
if( visible(p,level,exact) )
{
std::string line( prefix ) ;
line.append( "-" ) ;

View File

@ -35,7 +35,7 @@
namespace G
{
class GetOpt ;
} ;
}
// Class: G::GetOpt
// Description: A command line switch parser.
@ -45,6 +45,8 @@ class G::GetOpt
{
public:
typedef std::vector<std::string GAllocator(std::string) > StringArray ;
struct Level // Used by G::GetOpt for extra type safety.
{ unsigned int level ; explicit Level(unsigned int l) : level(l) {} } ;
G_EXCEPTION( InvalidSpecification , "invalid options specification string" ) ;
GetOpt( const Arg & arg , const std::string & spec ,
@ -58,9 +60,12 @@ public:
// <switch-description>
// <value-type> -- 0 is none, and 1 is a string
// <value-description>
// <level>
//
// If the switch-description field is empty
// then the switch is hidden.
// If the switch-description field is empty or
// if the level is zero then the switch is hidden.
// By convention main-stream switches should have
// a level of 1, and obscure ones level 2 and above.
Arg args() const ;
// Returns all the non-switch command-line arguments.
@ -71,28 +76,39 @@ public:
static size_t wrapDefault() ;
// Returns a default word-wrapping width.
static size_t tabDefault() ;
// Returns a default tab-stop.
static Level levelDefault() ;
// Returns the default level.
std::string usageSummary( const std::string & exe , const std::string & args ,
size_t wrap_width = wrapDefault() ) const ;
Level level = levelDefault() , size_t wrap_width = wrapDefault() ) const ;
// Returns a one-line usage summary, as
// "usage: <exe> <usageSummarySwitches()> <args>"
std::string usageSummarySwitches() const ;
std::string usageSummarySwitches( Level level = levelDefault() ) const ;
// Returns the one-line summary of switches. Does _not_
// include the usual "usage: <exe>" prefix
// or non-switch arguments.
std::string usageHelp( size_t tab_stop = 30U , size_t wrap_width = wrapDefault() ) const ;
// Returns a multi-line string giving help on each switch.
std::string usageHelp( Level level = levelDefault() ,
size_t tab_stop = tabDefault() , size_t wrap_width = wrapDefault() ,
bool level_exact = false ) const ;
// Returns a multi-line string giving help on each switch.
void showUsage( std::ostream & stream , const std::string & exe ,
const std::string & args , size_t tab_stop = 30U ,
const std::string & args , Level level = levelDefault() ,
size_t tab_stop = tabDefault() ,
size_t wrap_width = wrapDefault() ) const ;
// Streams out multi-line usage text using
// usageSummary() and usageHelp().
void showUsage( std::ostream & stream , const std::string & args ) const ;
void showUsage( std::ostream & stream , const std::string & args , bool verbose ) const ;
// Streams out multi-line usage text using
// usageSummary() and usageHelp().
// usageSummary() and usageHelp(). Shows
// only level one switches if 'verbose'
// is false.
bool hasErrors() const ;
// Returns true if there are errors.
@ -126,7 +142,7 @@ public:
// value-based switch.
private:
struct SwitchSpec
struct SwitchSpec // A private implementation structure used by G::GetOpt.
{
char c ;
std::string name ;
@ -134,11 +150,12 @@ private:
bool valued ;
bool hidden ;
std::string value_description ;
unsigned int level ;
SwitchSpec(char c_,const std::string &name_,const std::string &description_,
bool v_,const std::string &vd_) :
bool v_,const std::string &vd_,unsigned int level_) :
c(c_) , name(name_) , description(description_) ,
valued(v_) , hidden(description_.empty()) ,
value_description(vd_) {}
valued(v_) , hidden(description_.empty()||level_==0U) ,
value_description(vd_) , level(level_) {}
} ;
typedef std::map<std::string,SwitchSpec GLessAllocator(char,SwitchSpec) > SwitchSpecMap ;
typedef std::pair<bool,std::string> Value ;
@ -147,7 +164,8 @@ private:
void operator=( const GetOpt & ) ;
GetOpt( const GetOpt & ) ;
void parseSpec( const std::string & spec , char , char , char ) ;
void addSpec( const std::string & sort_key , char c , const std::string & name , const std::string & , bool valued , const std::string & ) ;
void addSpec( const std::string & , char c , const std::string & ,
const std::string & , bool , const std::string & , unsigned int ) ;
size_t parseArgs( const Arg & args_in ) ;
bool isOldSwitch( const std::string & arg ) const ;
bool isNewSwitch( const std::string & arg ) const ;
@ -166,10 +184,11 @@ private:
void remove( size_t n ) ;
bool valid( const std::string & ) const ;
bool valid( char c ) const ;
std::string usageSummaryPartOne() const ;
std::string usageSummaryPartTwo() const ;
std::string usageHelpCore( const std::string & , size_t , size_t ) const ;
std::string usageSummaryPartOne( Level ) const ;
std::string usageSummaryPartTwo( Level ) const ;
std::string usageHelpCore( const std::string & , Level , size_t , size_t , bool ) const ;
static size_t widthLimit( size_t ) ;
static bool visible( SwitchSpecMap::const_iterator , Level , bool ) ;
private:
SwitchSpecMap m_spec_map ;

View File

@ -28,7 +28,7 @@
namespace G
{
class LogImp ;
} ;
}
// Class: LogImp
// Description: An implementation class used by Log.
@ -36,22 +36,22 @@ namespace G
class G::LogImp
{
public:
static std::stringstream &s() ;
static std::ostringstream &s() ;
static bool active() ;
static void empty() ;
static const char *m_file ;
static int m_line ;
static std::stringstream *m_ss ;
static std::ostringstream *m_ss ;
} ;
const char *G::LogImp::m_file = NULL ;
std::stringstream *G::LogImp::m_ss = NULL ;
std::ostringstream *G::LogImp::m_ss = NULL ;
int G::LogImp::m_line = 0 ;
std::stringstream &G::LogImp::s()
std::ostringstream & G::LogImp::s()
{
if( m_ss == NULL )
m_ss = new std::stringstream ;
m_ss = new std::ostringstream ;
return *m_ss ;
}
@ -59,7 +59,7 @@ void G::LogImp::empty()
{
delete m_ss ;
m_ss = NULL ;
m_ss = new std::stringstream ;
m_ss = new std::ostringstream ;
}
bool G::LogImp::active()
@ -86,16 +86,7 @@ G::Log::End G::Log::end( G::Log::Severity severity )
G::Log::Stream & G::Log::stream()
{
if( G::LogImp::active() )
{
return G::LogImp::s() ;
}
else
{
static char buffer[3] ;
static std::stringstream dummy( buffer , sizeof(buffer) ) ;
return dummy ;
}
return G::LogImp::s() ;
}
void G::Log::onEnd( G::Log::Severity severity )
@ -104,10 +95,9 @@ void G::Log::onEnd( G::Log::Severity severity )
{
G::LogOutput::output( severity , G::LogImp::m_file , G::LogImp::m_line ,
G::LogImp::s().str().c_str() ) ;
// empty the stream
G::LogImp::empty() ;
}
G::LogImp::empty() ; // empty the stream
G::LogImp::m_file = NULL ;
G::LogImp::m_line = 0 ;
}

View File

@ -29,7 +29,7 @@
namespace G
{
class Log ;
} ;
}
// Class: G::Log
// Description: A static class for doing iostream-based logging.
@ -82,7 +82,7 @@ namespace G
{
return stream ;
}
} ;
}
namespace G
{
@ -93,7 +93,7 @@ namespace G
G::Log::onEnd( end.m_s ) ;
return stream ;
}
} ;
}
namespace G
{
@ -103,7 +103,7 @@ namespace G
G::Log::setFile( file ) ;
G::Log::setLine( line ) ;
}
} ;
}
// Macros: G_LOG, G_LOG_S, G_DEBUG, G_WARNING, G_ERROR
// The debug macro is for debugging during development. The log macro
@ -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 ) do { G::Log::stream() << G::Log::Line(__FILE__,__LINE__) << expr << G::Log::end(severity) ; } while(0)
#if defined(_DEBUG) && ! defined(G_NO_DEBUG)
#define G_DEBUG( expr ) G_LOG_OUTPUT( expr , G::Log::s_Debug )
#else

View File

@ -30,7 +30,7 @@
namespace G
{
class LogOutput ;
} ;
}
// Class: G::LogOutput
// Description: Controls and implements low-level logging output, as used by the Log interface.

View File

@ -46,7 +46,7 @@ namespace
return m ;
}
} ;
}
void G::LogOutput::rawOutput( G::Log::Severity severity , const char *message )
{

View File

@ -27,12 +27,6 @@
#include <cstdlib> // getenv
static HANDLE source() ;
#if ! defined(RegisterEventSource)
static void RegisterEventSource( const char * , const char * ) {}
static void DeregisterEventSource( HANDLE ) {}
static void ReportEvent( HANDLE , DWORD , int , DWORD , const char * ,
size_t , int , const char ** , const char * ) {}
#endif
void G::LogOutput::cleanup()
{
@ -90,9 +84,7 @@ void G::LogOutput::rawOutput( G::Log::Severity severity , const char *message )
void G::LogOutput::init()
{
#if defined(RegisterEventSource)
m_handle = ::source() ;
#endif
}
static HANDLE source()

View File

@ -31,7 +31,7 @@
namespace G
{
class Md5 ;
} ;
}
// Class: G::Md5
// Description: MD5 class.

View File

@ -29,7 +29,7 @@
namespace G
{
class noncopyable ;
} ;
}
// Class: G::noncopyable
// Description: A noncopyable base class (a la boost).

View File

@ -32,7 +32,7 @@
namespace G
{
class Path ;
} ;
}
// Class: G::Path
// Description: A Path object represents a file system
@ -86,7 +86,7 @@ public:
// eg. c:foo\bar.exe -> c:foo
// eg. c:\foo\bar.exe -> c:\foo
// eg. c:bar.exe -> c:
// eg. c:\file -> c:\
// eg. c:\file -> c:\ .
// eg. c:\ -> <empty>
// eg. c: -> <empty>
// eg. \foo\bar.exe -> \foo
@ -105,8 +105,8 @@ public:
void removeExtension() ;
// Modifies the path by removing any extension.
// However, the extension returned by extension()
// is unchanged.
// However, the extension returned by extension() is
// unchanged.
void setExtension( const std::string & extension ) ;
// Replaces the extension. Any leading dot in the
@ -177,6 +177,6 @@ namespace G
path.streamOut( stream ) ;
return stream ;
}
} ;
}
#endif

128
src/glib/gpidfile.cpp Normal file
View File

@ -0,0 +1,128 @@
//
// Copyright (C) 2001-2002 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.
//
// ===
//
// gpidfile.cpp
//
#include "gdef.h"
#include "gpidfile.h"
#include "gprocess.h"
#include "gcleanup.h"
#include "gfile.h"
#include "gdebug.h"
#include <cstdlib> // std::malloc()
#include <cstring> // std::strdup()
#include <fstream>
#include <fcntl.h>
namespace
{
// strdup() not in std:: ?
char * strdup_( const char * p )
{
p = p ? p : "" ;
char * buffer = static_cast<char*>( std::malloc( std::strlen(p) + 1U ) ) ;
std::strcpy( buffer , p ) ;
return buffer ;
}
}
//static
void G::PidFile::create( const Path & pid_file )
{
if( pid_file != Path() )
{
G_DEBUG( "G::PidFile::create: \"" << pid_file << "\"" ) ;
Process::Umask readable(Process::Umask::Readable) ;
std::ofstream file( pid_file.str().c_str() ) ;
file << Process::Id() << std::endl ;
if( !file.good() )
throw Error(std::string("cannot create file: ")+pid_file.str()) ;
Cleanup::add( cleanup , strdup_(pid_file.str().c_str()) ) ; // (leaks)
}
}
//static
bool G::PidFile::mine( const char * path )
{
// reentrant implementation...
Process::Id this_pid ;
Process::Id file_pid( path ) ;
return this_pid == file_pid ;
}
//static
void G::PidFile::cleanup( const char * path )
{
// reentrant implementation...
try
{
if( path && *path && mine(path) )
std::remove( path ) ;
}
catch(...)
{
}
}
// ===
G::PidFile::PidFile()
{
}
G::PidFile::~PidFile()
{
if( valid() )
cleanup( m_path.pathCstr() ) ;
}
G::PidFile::PidFile( const Path & path ) :
m_path(path)
{
}
void G::PidFile::init( const Path & path )
{
m_path = path ;
}
void G::PidFile::check()
{
if( valid() && ! m_path.isAbsolute() )
throw Error(std::string("must be an absolute path: ")+m_path.str()) ;
}
void G::PidFile::commit()
{
if( valid() )
create( m_path ) ;
}
G::Path G::PidFile::path() const
{
return m_path ;
}
bool G::PidFile::valid() const
{
return m_path != Path() ;
}

93
src/glib/gpidfile.h Normal file
View File

@ -0,0 +1,93 @@
//
// Copyright (C) 2001-2002 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.
//
// ===
//
// gpidfile.h
//
#ifndef G_PIDFILE_H
#define G_PIDFILE_H
#include "gdef.h"
#include "gexception.h"
#include "gpath.h"
#include <sys/types.h>
#include <string>
namespace G
{
class PidFile ;
class Daemon ;
}
// Class: G::PidFile
// Description: A class for creating pid files.
// See also: G::Daemon
//
class G::PidFile
{
public:
G_EXCEPTION( Error , "invalid pid file" ) ;
static void cleanup( const char * path ) ;
// Deletes the specified pid file if it
// contains this process's id.
//
// Reentrant implementation.
explicit PidFile( const Path & pid_file_path ) ;
// Constructor. The path should normally be
// an absolute path. Use commit() to actually
// create the file.
PidFile() ;
// Default constructor. Constructs a
// do-nothing object. Initialise with init().
void init( const Path & pid_file_path ) ;
// Used after default construction.
~PidFile() ;
// Destructor. Calls cleanup() to delete
// the file.
void commit() ;
// Creates the file.
void check() ;
// Throws an exception if the path is not
// absolute.
Path path() const ;
// Returns the path as supplied to the constructor
// or init().
private:
Path m_path ;
private:
PidFile( const PidFile & ) ; // not implemented
void operator=( const PidFile & ) ; // not implemented
static bool mine( const char * path ) ; // reentrant
static void create( const Path & pid_file ) ;
bool valid() const ;
} ;
#endif

View File

@ -27,6 +27,7 @@
#include "gdef.h"
#include "gexception.h"
#include "gpath.h"
#include "gstrings.h"
#include <iostream>
#include <sys/types.h>
#include <string>
@ -34,7 +35,7 @@
namespace G
{
class Process ;
} ;
}
// Class: G::Process
// Description: A static interface for doing things with processes.
@ -52,18 +53,19 @@ public:
G_EXCEPTION( UidError , "cannot set uid" ) ;
G_EXCEPTION( GidError , "cannot set gid" ) ;
G_EXCEPTION( Insecure , "refusing to exec() while the user-id is zero" ) ;
G_EXCEPTION( InvalidId , "invalid process-id string" ) ;
G_EXCEPTION( PipeError , "pipe error" ) ;
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: explicit Id( std::istream & ) ;
public: explicit Id( const char * pid_file_path ) ; // (re-entrant ctor)
public: std::string str() const ;
private: IdImp * m_imp ;
public: bool operator==( const Id & ) const ;
private: pid_t m_pid ;
friend class Process ;
} ;
struct Identity // Used by G::Process::beSpecial().
@ -74,6 +76,17 @@ public:
explicit Identity( const std::string & login_name ) ;
std::string str() const ;
} ;
class Umask // Used to temporarily modify the process umask.
{
public: enum Mode { Readable , Tighter , Tightest } ;
public: explicit Umask( Mode ) ;
public: ~Umask() ;
public: static void set( Mode ) ;
private: Umask( const Umask & ) ; // not implemented
private: void operator=( const Umask & ) ; // not implemented
private: class UmaskImp ;
private: UmaskImp * m_imp ;
} ;
class NoThrow // An overload discriminator for Process.
{} ;
@ -83,9 +96,6 @@ public:
static void closeStderr() ;
// Closes stderr.
static void setUmask( bool tightest = true ) ;
// Sets a tight umask.
static void cd( const Path & dir ) ;
// Changes directory.
@ -98,12 +108,15 @@ public:
static Who fork( Id & child ) ;
// Forks a child process. Returns the child
// pid to the parent.
// pid by reference to the parent.
static int spawn( Identity nobody , const Path & exe , const std::string & arg , int error_return = 127 ) ;
// Runs a command in an unprivileged child process. Returns the
// child process's exit code, or 'error_return' on error.
// The identity should have come from beOrdinary().
static int spawn( Identity nobody , const Path & exe , const Strings & args ,
std::string * pipe_result_p = NULL , int error_return = 127 ) ;
// Runs a command in an unprivileged child process. Returns the
// child process's exit code, or 'error_return' on error.
// If the 'pipe_result_p' pointer is supplied then a pipe
// is used to read the first bit of whatever the child process
// writes to stdout. The identity should have come from beOrdinary().
static int errno_() ;
// Returns the process's current 'errno' value.
@ -112,6 +125,8 @@ public:
// Aquires special privileges (either root
// or suid). The parameter must have come from
// a previous call to beOrdinary().
//
// See also class G::Root.
static Identity beOrdinary( Identity nobody , bool change_group = true ) ;
// Revokes special privileges (root or suid).
@ -122,13 +137,16 @@ public:
// parameter is ignored. Returns the old
// identity, which can be passed to
// beSpecial().
//
// See also class G::Root.
private:
Process() ;
static int wait( const Id & child ) ;
static int wait( const Id & child , int error_return ) ;
static void execCore( const Path & , const std::string & ) ;
static void execCore( const Path & , const Strings & ) ;
static void beNobody( Identity ) ;
static void closeFiles( int ) ;
} ;
namespace G
@ -139,12 +157,19 @@ namespace G
return stream << id.str() ;
}
inline
std::istream & operator>>( std::istream & stream , G::Process::Id & id )
{
id = G::Process::Id( stream ) ;
return stream ;
}
inline
std::ostream & operator<<( std::ostream & stream , const G::Process::Identity & identity )
{
return stream << identity.str() ;
}
} ;
}
#endif

View File

@ -34,6 +34,38 @@
#include <pwd.h> // getpwnam()
#include <unistd.h> // setuid() etc
namespace
{
void noCloseOnExec( int fd )
{
::fcntl( fd , F_SETFD , 0 ) ;
}
}
namespace G
{
class Pipe ;
}
// Class: G::Pipe
// Description: A private implementation class used by G::Process.
//
class G::Pipe
{
public:
explicit Pipe( bool active ) ;
~Pipe() ;
void inChild() ; // writer
void inParent() ; // reader
int fd() const ;
void dup() ; // onto stdout
std::string read() ; // size-limited
private:
G_EXCEPTION( Error , "pipe error" ) ;
int m_fds[2] ;
int m_fd ;
} ;
// Class: G::Process::IdImp
// Description: A private implementation class used by G::Process.
//
@ -58,24 +90,25 @@ bool G::Process::cd( const Path & dir , NoThrow )
return 0 == ::chdir( dir.str().c_str() ) ;
}
//static
void G::Process::setUmask( bool tightest )
{
mode_t new_mode = tightest ? 0177 : 0117 ; // tightest => -rw-------
(void) ::umask( new_mode ) ;
}
//static
void G::Process::closeStderr()
{
::close( STDERR_FILENO ) ;
::open( G::FileSystem::nullDevice() , O_WRONLY ) ;
::fcntl( STDERR_FILENO , F_SETFD , 0 ) ; // close-on-exec false
noCloseOnExec( STDERR_FILENO ) ;
}
//static
void G::Process::closeFiles( bool keep_stderr )
{
closeFiles( keep_stderr ? STDERR_FILENO : -1 ) ;
}
//static
void G::Process::closeFiles( int keep )
{
G_ASSERT( keep == -1 || keep >= STDERR_FILENO ) ;
int n = 256U ;
long rc = ::sysconf( _SC_OPEN_MAX ) ;
if( rc > 0L )
@ -83,10 +116,8 @@ void G::Process::closeFiles( bool keep_stderr )
for( int fd = 0 ; fd < n ; fd++ )
{
if( !keep_stderr || fd != STDERR_FILENO )
{
if( fd != keep )
::close( fd ) ;
}
}
// reopen standard fds to prevent accidental use
@ -95,13 +126,13 @@ void G::Process::closeFiles( bool keep_stderr )
//
::open( G::FileSystem::nullDevice() , O_RDONLY ) ;
::open( G::FileSystem::nullDevice() , O_WRONLY ) ;
if( !keep_stderr )
if( keep != STDERR_FILENO )
{
::open( G::FileSystem::nullDevice() , O_WRONLY ) ;
::fcntl( STDERR_FILENO , F_SETFD , 0 ) ; // close-on-exec false
}
::fcntl( STDIN_FILENO , F_SETFD , 0 ) ; // close-on-exec false
::fcntl( STDOUT_FILENO , F_SETFD , 0 ) ; // close-on-exec false
noCloseOnExec( STDIN_FILENO ) ;
noCloseOnExec( STDOUT_FILENO ) ;
noCloseOnExec( STDERR_FILENO ) ;
}
G::Process::Who G::Process::fork()
@ -117,7 +148,7 @@ G::Process::Who G::Process::fork( Id & child_pid )
if( ok )
{
if( rc != 0 )
child_pid.m_imp->m_pid = rc ;
child_pid.m_pid = rc ;
}
else
{
@ -132,7 +163,7 @@ int G::Process::wait( const Id & child_pid )
for(;;)
{
G_DEBUG( "G::Process::wait: waiting" ) ;
int rc = ::waitpid( child_pid.m_imp->m_pid , &status , 0 ) ;
int rc = ::waitpid( child_pid.m_pid , &status , 0 ) ;
if( rc == -1 && errno_() == EINTR )
{
; // signal in parent -- keep waiting
@ -140,7 +171,9 @@ int G::Process::wait( const Id & child_pid )
else if( rc == -1 )
{
int error = errno_() ;
throw WaitError( std::stringstream() << "errno=" << error ) ;
std::ostringstream ss ;
ss << "errno=" << error ;
throw WaitError( ss.str() ) ;
}
else
{
@ -152,7 +185,9 @@ int G::Process::wait( const Id & child_pid )
if( ! WIFEXITED(status) )
{
// uncaught signal or stopped
throw ChildError( std::stringstream() << "status=" << status ) ;
std::ostringstream ss ;
ss << "status=" << status ;
throw ChildError( ss.str() ) ;
}
const int exit_status = WEXITSTATUS(status) ;
@ -177,7 +212,8 @@ int G::Process::errno_()
return errno ; // not ::errno or std::errno for gcc2.95
}
int G::Process::spawn( Identity nobody , const G::Path & exe , const std::string & arg , int error_return )
int G::Process::spawn( Identity nobody , const Path & exe , const Strings & args ,
std::string * pipe_result_p , int error_return )
{
if( exe.isRelative() )
throw InvalidPath( exe.str() ) ;
@ -185,28 +221,36 @@ int G::Process::spawn( Identity nobody , const G::Path & exe , const std::string
if( ::geteuid() == 0U || nobody.uid == 0U )
throw Insecure() ;
Pipe pipe( pipe_result_p != NULL ) ;
Id child_pid ;
if( fork(child_pid) == Child )
{
beNobody( nobody ) ;
G_ASSERT( ::getuid() != 0U && ::geteuid() != 0U ) ;
closeFiles() ;
execCore( exe , arg ) ;
try
{
beNobody( nobody ) ;
G_ASSERT( ::getuid() != 0U && ::geteuid() != 0U ) ;
pipe.inChild() ;
closeFiles( pipe.fd() ) ;
pipe.dup() ; // dup() onto stdout
execCore( exe , args ) ;
}
catch(...)
{
}
::_exit( error_return ) ;
return error_return ; // pacify the compiler
}
else
{
return wait( child_pid , error_return ) ;
pipe.inParent() ;
int exit_status = wait( child_pid , error_return ) ;
if( pipe_result_p != NULL ) *pipe_result_p = pipe.read() ;
return exit_status ;
}
}
void G::Process::execCore( const G::Path & exe , const std::string & arg )
void G::Process::execCore( const G::Path & exe , const Strings & args )
{
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 ;
char * env[3U] ;
std::string path( "PATH=/usr/bin:/bin" ) ; // no "."
std::string ifs( "IFS= \t\n" ) ;
@ -214,9 +258,17 @@ void G::Process::execCore( const G::Path & exe , const std::string & arg )
env[1U] = const_cast<char*>( ifs.c_str() ) ;
env[2U] = NULL ;
::execve( exe.str().c_str() , argv , env ) ;
char ** argv = new char* [ args.size() + 2U ] ;
argv[0U] = const_cast<char*>( exe.pathCstr() ) ;
unsigned int argc = 1U ;
for( Strings::const_iterator arg_p = args.begin() ; arg_p != args.end() ; ++arg_p , argc++ )
argv[argc] = const_cast<char*>(arg_p->c_str()) ;
argv[argc] = NULL ;
::execve( exe.str().c_str() , argv , env ) ;
const int error = errno_() ;
delete [] argv ;
G_WARNING( "G::Process::exec: execve() returned: errno=" << error << ": " << exe ) ;
}
@ -232,8 +284,8 @@ void G::Process::beSpecial( Identity identity , bool change_group )
Identity old_identity ;
(void) ::seteuid( identity.uid ) ;
if( change_group) (void) ::setegid( identity.gid ) ;
(void) old_identity.str() ; // pacify the compiler
G_DEBUG( "G::Process::beSpecial: " << old_identity << " -> " << Identity() ) ;
//G_DEBUG( "G::Process::beSpecial: " << old_identity << " -> " << Identity() ) ;
old_identity.uid = 0 ; // pacify the compiler
}
G::Process::Identity G::Process::beOrdinary( Identity nobody , bool change_group )
@ -252,7 +304,7 @@ G::Process::Identity G::Process::beOrdinary( Identity nobody , bool change_group
if( ::seteuid( ::getuid() ) ) throw UidError() ;
if( change_group && ::setegid( ::getgid() ) ) throw GidError() ;
}
G_DEBUG( "G::Process::beOrdinary: " << special_identity << " -> " << Identity() ) ;
//G_DEBUG( "G::Process::beOrdinary: " << special_identity << " -> " << Identity() ) ;
return special_identity ;
}
@ -269,40 +321,42 @@ void G::Process::beNobody( Identity nobody )
// ===
G::Process::Id::Id() : m_imp(NULL)
G::Process::Id::Id()
{
m_imp = new IdImp ;
m_imp->m_pid = ::getpid() ;
m_pid = ::getpid() ;
}
G::Process::Id::~Id()
G::Process::Id::Id( const char * path ) :
m_pid(0)
{
delete m_imp ;
// reentrant implementation suitable for a signal handler...
int fd = ::open( path ? path : "" , O_RDONLY ) ;
char buffer[10] ;
ssize_t rc = ::read( fd , buffer , sizeof(buffer) ) ;
for( const char * p = buffer ; rc > 0 && *p >= '0' && *p <= '9' ; p++ , rc-- )
{
m_pid *= 10 ;
m_pid += ( *p - '0' ) ;
}
}
G::Process::Id::Id( const Id & other ) :
m_imp(NULL)
G::Process::Id::Id( std::istream & stream )
{
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 ;
stream >> m_pid ;
if( !stream.good() )
throw Process::InvalidId() ;
}
std::string G::Process::Id::str() const
{
std::stringstream ss ;
ss << m_imp->m_pid ;
std::ostringstream ss ;
ss << m_pid ;
return ss.str() ;
}
bool G::Process::Id::operator==( const Id & rhs ) const
bool G::Process::Id::operator==( const Id & other ) const
{
return m_imp->m_pid == rhs.m_imp->m_pid ;
return m_pid == other.m_pid ;
}
// ===
@ -322,7 +376,7 @@ G::Process::Identity::Identity( const std::string & name ) :
::passwd * pw = ::getpwnam( name.c_str() ) ;
if( pw == NULL )
throw Process::NoSuchUser(name) ;
G_DEBUG( "G::Process::Identity: " << name << "=" << pw->pw_uid << "/" << pw->pw_gid ) ;
//G_DEBUG( "G::Process::Identity: " << name << "=" << pw->pw_uid << "/" << pw->pw_gid ) ;
uid = pw->pw_uid ;
gid = pw->pw_gid ;
}
@ -330,8 +384,95 @@ G::Process::Identity::Identity( const std::string & name ) :
std::string G::Process::Identity::str() const
{
std::stringstream ss ;
std::ostringstream ss ;
ss << uid << "/" << gid ;
return ss.str() ;
}
// ===
class G::Process::Umask::UmaskImp
{
public:
mode_t m_old_mode ;
} ;
G::Process::Umask::Umask( Mode mode ) :
m_imp(new UmaskImp)
{
m_imp->m_old_mode =
::umask( mode==Readable?0133:(mode==Tighter?0117:0177) ) ;
}
G::Process::Umask::~Umask()
{
(void) ::umask( m_imp->m_old_mode ) ;
delete m_imp ;
}
//static
void G::Process::Umask::set( Mode mode )
{
// Tightest: -rw-------
// Tighter: -rw-rw----
// Readable: -rw-r--r--
::umask( mode==Readable?0133:(mode==Tighter?0117:0177) ) ;
}
// ===
G::Pipe::Pipe( bool active ) :
m_fd(-1)
{
m_fds[0] = m_fds[1] = -1 ;
if( active && ::pipe( m_fds ) < 0 )
throw Error() ;
G_DEBUG( "G::Pipe::ctor: " << m_fds[0] << " " << m_fds[1] ) ;
}
G::Pipe::~Pipe()
{
if( m_fds[0] >= 0 ) ::close( m_fds[0] ) ;
if( m_fds[1] >= 0 ) ::close( m_fds[1] ) ;
}
void G::Pipe::inChild()
{
::close( m_fds[0] ) ;
m_fds[0] = -1 ;
m_fd = m_fds[1] ; // writer
}
void G::Pipe::inParent()
{
::close( m_fds[1] ) ;
m_fds[1] = -1 ;
m_fd = m_fds[0] ; // reader
}
int G::Pipe::fd() const
{
return m_fd ;
}
void G::Pipe::dup()
{
if( m_fd != -1 && m_fd != STDOUT_FILENO )
{
if( ::dup2(m_fd,STDOUT_FILENO) != STDOUT_FILENO )
throw Error() ;
::close( m_fd ) ;
m_fd = -1 ;
m_fds[1] = -1 ;
noCloseOnExec( STDOUT_FILENO ) ;
}
}
std::string G::Pipe::read()
{
char buffer[4096] ;
int rc = m_fd == -1 ? 0 : ::read( m_fd , buffer , sizeof(buffer) ) ;
if( rc < 0 ) throw Error("read") ;
return std::string(buffer,rc) ;
}

View File

@ -23,20 +23,36 @@
#include "gdef.h"
#include "gprocess.h"
#include "gexception.h"
#include "gstr.h"
#include "glog.h"
#include <sys/types.h>
#include <sys/stat.h>
#include <process.h>
#include <direct.h>
#include <io.h>
#include <fcntl.h>
namespace G
{
const int STDERR_FILENO = 2 ;
const int SC_OPEN_MAX = 256 ; // 32 in limits.h !?
class Pipe ;
} ;
// ===
class G::Pipe
{
public:
G_EXCEPTION( Error , "pipe error" ) ;
explicit Pipe( bool active ) ;
~Pipe() ;
int fd() const ;
std::string read() ;
private:
bool m_active ;
int m_fds[2] ;
int m_fd_writer ;
} ;
class G::Process::IdImp
{
@ -46,42 +62,86 @@ public:
// ===
G::Process::Id::Id() : m_imp(NULL)
G::Pipe::Pipe( bool active ) :
m_active(active) ,
m_fd_writer(-1)
{
m_imp = new IdImp ;
m_imp->m_pid = static_cast<unsigned int>(::_getpid()) ; // or ::GetCurrentProcessId()
m_fds[0] = m_fds[1] = -1 ;
if( m_active )
{
int rc = ::_pipe( m_fds , 256 , _O_BINARY | _O_NOINHERIT ) ;
if( rc < 0 ) throw Error() ;
m_fd_writer = ::_dup( m_fds[1] ) ; // inherited
::_close( m_fds[1] ) ;
m_fds[1] = -1 ;
}
}
G::Process::Id::~Id()
G::Pipe::~Pipe()
{
delete m_imp ;
if( m_active )
{
::_close( m_fds[0] ) ;
::_close( m_fds[1] ) ;
::_close( m_fd_writer ) ;
}
}
G::Process::Id::Id( const Id & other ) :
m_imp(NULL)
int G::Pipe::fd() const
{
m_imp = new IdImp ;
m_imp->m_pid = other.m_imp->m_pid ;
return m_fd_writer ;
}
G::Process::Id & G::Process::Id::operator=( const Id & rhs )
std::string G::Pipe::read()
{
m_imp->m_pid = rhs.m_imp->m_pid ;
return *this ;
if( ! m_active ) return std::string() ;
::_close( m_fd_writer ) ;
char buffer[4096] ;
int rc = m_fds[0] == -1 ? 0 : ::_read( m_fds[0] , buffer , sizeof(buffer) ) ;
if( rc < 0 ) throw Error() ;
return std::string( buffer , rc ) ;
}
// ===
G::Process::Id::Id()
{
m_pid = static_cast<unsigned int>(::_getpid()) ; // or ::GetCurrentProcessId()
}
G::Process::Id::Id( const char * path ) :
m_pid(0)
{
std::ifstream file( path ? path : "" ) ;
file >> m_pid ;
if( !file.good() )
m_pid = 0 ;
}
G::Process::Id::Id( std::istream & stream )
{
stream >> m_pid ;
if( !stream.good() )
throw Process::InvalidId() ;
}
std::string G::Process::Id::str() const
{
std::stringstream ss ;
ss << m_imp->m_pid ;
std::ostringstream ss ;
ss << m_pid ;
return ss.str() ;
}
bool G::Process::Id::operator==( const Id & rhs ) const
{
return m_imp->m_pid == rhs.m_imp->m_pid ;
return m_pid == rhs.m_pid ;
}
// not implemented...
//G::Process::Id::Id( const char * pid_file_path ) {}
// ===
void G::Process::closeFiles( bool keep_stderr )
{
const int n = SC_OPEN_MAX ;
@ -92,12 +152,6 @@ void G::Process::closeFiles( bool keep_stderr )
}
}
void G::Process::setUmask( bool )
{
// _umask() is available but not very useful
; // no-op
}
void G::Process::closeStderr()
{
int fd = STDERR_FILENO ;
@ -115,21 +169,33 @@ bool G::Process::cd( const Path & dir , NoThrow )
return 0 == ::_chdir( dir.str().c_str() ) ;
}
int G::Process::spawn( Identity , const Path & exe , const std::string & arg ,
int error_return )
int G::Process::spawn( Identity , const Path & exe , const Strings & args_ ,
std::string * pipe_result_p , 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 ;
Strings args( args_ ) ; // non-const copy
Pipe pipe( pipe_result_p != NULL ) ;
if( pipe_result_p != NULL )
args.push_front( Str::fromInt(pipe.fd()) ) ; // kludge -- child must write on specific fd passed as argv[1]
char ** argv = new char* [ args.size() + 2U ] ;
argv[0U] = const_cast<char*>( exe.pathCstr() ) ;
unsigned int argc = 1U ;
for( Strings::const_iterator arg_p = args.begin() ; arg_p != args.end() ; ++arg_p , argc++ )
argv[argc] = const_cast<char*>(arg_p->c_str()) ;
argv[argc] = NULL ;
const int mode = _P_WAIT ;
::_flushall() ;
int rc = ::_spawnv( mode , exe.str().c_str() , argv ) ;
delete [] argv ;
if( pipe_result_p != NULL )
*pipe_result_p = pipe.read() ;
return rc < 0 ? error_return : rc ;
}
@ -171,3 +237,19 @@ std::string G::Process::Identity::str() const
return "0/0" ;
}
// ===
G::Process::Umask::Umask( G::Process::Umask::Mode ) :
m_imp(0)
{
}
G::Process::Umask::~Umask()
{
}
void G::Process::Umask::set( G::Process::Umask::Mode )
{
// not implemented
}

168
src/glib/gregistry.h Normal file
View File

@ -0,0 +1,168 @@
//
// Copyright (C) 2001-2002 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.
//
// ===
//
// gregistry.h
//
#ifndef G_REGISTRY_H
#define G_REGISTRY_H
#include "gdef.h"
#include "gexception.h"
#include <string>
namespace G
{
class RegistryKey ;
class RegistryKeyImp ;
class RegistryValue ;
}
// Class: G::RegistryKey
// Description: Used to navigate the system registry. Works with
// the G::RegistryValue class to get and set values.
// See also: G::RegistryValue
//
class G::RegistryKey
{
public:
G_EXCEPTION( InvalidHandle , "registry handle error" ) ;
G_EXCEPTION( RemoveError , "registry removal error" ) ;
G_EXCEPTION( Error , "registry error" ) ;
struct NoThrow // Overload discriminator for G::RegistryKey.
{} ;
static RegistryKey currentUser() ;
// Returns a key for the current-user "hive".
static RegistryKey localMachine() ;
// Returns a key for the local-machine "hive".
static RegistryKey classes() ;
// Returns a key for the classes-root "hive".
~RegistryKey() ;
// Destructor.
RegistryKey create( std::string sub_path , bool & created ) const ;
// Opens or creates a sub-key.
RegistryKey create( const std::string & sub_path ) const ;
// Opens or creates a sub-key.
RegistryKey open( const std::string & sub_path ) const ;
// Opens an existing sub-key. Throws if non-existant.
RegistryKey open( const std::string & sub_path , const NoThrow & ) const ;
// Opens an existing sub-key. Returns
// an invalid key on error (eg. if non-existant).
bool valid() const ;
// Returns true if a valid key.
// (Invalid keys are only created
// by the NoThrow overload of
// open().)
RegistryKey( const RegistryKey & ) ;
// Copy ctor.
void operator=( const RegistryKey & ) ;
// Assignment operator.
void remove( const std::string & sub_path ) const ;
// Removes the named sub-key.
// Throws on error.
void remove( const std::string & sub_path , const NoThrow & ) const ;
// Removes the named sub-key.
// Ignores errors.
const RegistryKeyImp & imp() const ;
// Used by RegistryValue.
private:
typedef RegistryKeyImp Imp ;
RegistryKey( const Imp & , bool ) ;
void down() ;
RegistryKey open( std::string , bool ) const ;
void remove( const std::string & , bool ) const ;
private:
Imp * m_imp ;
bool m_is_root ;
} ;
// Class: G::RegistryValue
// Description: Works with G::RegistryKey to get and set
// registry values.
// See also: G::RegistryKey
//
class G::RegistryValue
{
public:
G_EXCEPTION( InvalidHandle , "registry handle error" ) ;
G_EXCEPTION( ValueError , "registry value error" ) ;
G_EXCEPTION( InvalidType , "registry type error" ) ;
G_EXCEPTION( MissingValue , "missing registry value" ) ;
explicit RegistryValue( const RegistryKey & hkey ,
const std::string & name = std::string() ) ;
// Constructor.
std::string getString() const ;
// Returns a string value. Throws if the value
// does not exist, or if it is not a string type.
std::string getString( const std::string & defolt ) const ;
// Returns a string. Returns the supplied default
// value if it does not exist.
bool getBool() const ;
// Returns a boolean value. Throws if the value
// does not exist.
g_uint32_t getDword() const ;
// Returns an unsigned 32-bit value. Throws if the value
// does not exist.
void set( const std::string & ) ;
// Stores a string value.
void set( const char * ) ;
// Stores a string value.
void set( bool ) ;
// Stores a boolean value.
void set( g_uint32_t ) ;
// Stores an unsigned 32-bit value.
private:
std::string getString( const std::string & defolt , bool ) const ;
std::pair<g_uint32_t,size_t> getInfo() const ;
std::string getData( g_uint32_t & type , bool & ) const ;
size_t get( char * , size_t ) const ;
private:
const RegistryKey & m_hkey ;
std::string m_key_name ;
} ;
#endif

View File

@ -0,0 +1,347 @@
//
// Copyright (C) 2001-2002 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.
//
// ===
//
// gregistry_win32.cpp
//
#include "gdef.h"
#include "gregistry.h"
#include "gstr.h"
#include "gpath.h"
#include "gdebug.h"
#include "gassert.h"
#include "glog.h"
#include <sstream>
class G::RegistryKeyImp
{
public:
HKEY m_key ;
unsigned long m_ref_count ;
explicit RegistryKeyImp( HKEY k ) : m_key(k) , m_ref_count(1UL) {}
} ;
// ===
G::RegistryKey::RegistryKey( const Imp & imp , bool is_root ) :
m_imp( new Imp(imp) ) ,
m_is_root(is_root)
{
m_imp->m_ref_count = 1UL ;
}
G::RegistryKey::RegistryKey( const RegistryKey & other ) :
m_imp( other.m_imp ) ,
m_is_root( other.m_is_root )
{
m_imp->m_ref_count++ ;
}
bool G::RegistryKey::valid() const
{
return m_imp->m_key != 0 ;
}
void G::RegistryKey::operator=( const RegistryKey & other )
{
down() ;
m_imp = other.m_imp ;
m_imp->m_ref_count++ ;
}
const G::RegistryKeyImp & G::RegistryKey::imp() const
{
return *m_imp ;
}
G::RegistryKey::~RegistryKey()
{
down() ;
}
void G::RegistryKey::down()
{
m_imp->m_ref_count-- ;
if( m_imp->m_ref_count == 0U && !m_is_root && m_imp->m_key != 0 )
{
G_DEBUG( "G::RegistryKey::down: closing " << m_imp->m_key ) ;
::RegCloseKey( m_imp->m_key ) ;
delete m_imp ;
m_imp = NULL ;
}
}
G::RegistryKey G::RegistryKey::create( const std::string & path ) const
{
bool is_new = false ;
return create( path , is_new ) ;
}
G::RegistryKey G::RegistryKey::create( std::string path , bool & is_new ) const
{
if( !valid() ) throw InvalidHandle() ;
G::Str::replaceAll( path , "/" , "\\" ) ;
is_new = false ;
HKEY new_key = 0 ;
DWORD disposition = 0 ;
LONG rc = ::RegCreateKeyEx( m_imp->m_key , path.c_str() , 0 ,
NULL , // ???
REG_OPTION_NON_VOLATILE ,
KEY_ALL_ACCESS , NULL ,
&new_key ,
&disposition ) ;
if( rc != ERROR_SUCCESS )
{
if( new_key != 0 )
::RegCloseKey(new_key) ;
G_DEBUG( "G::RegistryKey::create: failed to create \"" << path << "\"" ) ;
throw Error( path ) ;
}
is_new = disposition == REG_CREATED_NEW_KEY ;
G_DEBUG( "G::RegistryKey::create: " << new_key << ": \"" << path << "\"" << (is_new?" [created]":"") ) ;
return RegistryKey( Imp(new_key) , false ) ;
}
G::RegistryKey G::RegistryKey::open( const std::string & path , const NoThrow & ) const
{
return open( path , false ) ;
}
G::RegistryKey G::RegistryKey::open( const std::string & path ) const
{
return open( path , true ) ;
}
G::RegistryKey G::RegistryKey::open( std::string path , bool do_throw ) const
{
if( !valid() )
{
if( do_throw )
throw InvalidHandle() ;
G_DEBUG( "G::RegistryKey::open: failed to open \"" << path << "\"" ) ;
return RegistryKey( Imp(0) , false ) ;
}
else
{
G::Str::replaceAll( path , "/" , "\\" ) ;
HKEY new_key = 0 ;
LONG rc = ::RegOpenKeyEx( m_imp->m_key , path.c_str() , 0 ,
KEY_ALL_ACCESS , &new_key ) ;
G_ASSERT( (rc == ERROR_SUCCESS) == (new_key != 0) ) ;
if( rc == ERROR_SUCCESS )
{
G_DEBUG( "G::RegistryKey::open: " << new_key << ": \"" << path << "\"" ) ;
}
else
{
if( new_key != 0 ) ::RegCloseKey(new_key) ;
if( do_throw ) throw Error( path ) ;
new_key = 0 ;
G_DEBUG( "G::RegistryKey::open: failed to open \"" << path << "\"" ) ;
}
return RegistryKey( Imp(new_key) , false ) ;
}
}
void G::RegistryKey::remove( const std::string & sub_key , const NoThrow & ) const
{
remove( sub_key , false ) ;
}
void G::RegistryKey::remove( const std::string & sub_key ) const
{
remove( sub_key , true ) ;
}
void G::RegistryKey::remove( const std::string & sub_key , bool do_throw ) const
{
if( !valid() )
{
if( do_throw )
throw InvalidHandle() ;
G_DEBUG( "G::RegistryKey::remove: failed to remove \"" << sub_key << "\"" ) ;
}
else
{
LONG rc = ::RegDeleteKey( m_imp->m_key , sub_key.c_str() ) ;
if( rc == ERROR_SUCCESS )
{
G_DEBUG( "G::RegistryKey::remove: removed \"" << sub_key << "\"" ) ;
}
else
{
if( do_throw )
throw RemoveError( sub_key ) ;
G_DEBUG( "G::RegistryKey::remove: failed to remove \"" << sub_key << "\"" ) ;
}
}
}
G::RegistryKey G::RegistryKey::currentUser()
{
return RegistryKey( Imp(HKEY_CURRENT_USER) , true ) ;
}
G::RegistryKey G::RegistryKey::localMachine()
{
return RegistryKey( Imp(HKEY_LOCAL_MACHINE) , true ) ;
}
G::RegistryKey G::RegistryKey::classes()
{
return RegistryKey( Imp(HKEY_CLASSES_ROOT) , true ) ;
}
// ===
G::RegistryValue::RegistryValue( const RegistryKey & hkey ,
const std::string & key_name ) :
m_hkey(hkey) ,
m_key_name(key_name)
{
if( ! hkey.valid() )
throw InvalidHandle( key_name ) ;
}
std::string G::RegistryValue::getString( const std::string & defolt ) const
{
return getString( defolt , false ) ;
}
std::string G::RegistryValue::getString() const
{
return getString( std::string() , true ) ;
}
std::string G::RegistryValue::getString( const std::string & defolt ,
bool do_throw ) const
{
DWORD type = 0 ;
bool exists = true ;
std::string result = getData( type , exists ) ;
if( !exists && do_throw )
{
throw MissingValue( m_key_name ) ;
}
else if( !exists )
{
return defolt ;
}
else if( type != REG_SZ && type != REG_EXPAND_SZ )
{
throw InvalidType( m_key_name ) ;
}
else
{
G::Str::trimRight( result , std::string(1U,'\0') ) ;
return result ;
}
}
std::string G::RegistryValue::getData( g_uint32_t & type , bool & exists ) const
{
exists = true ;
std::pair<DWORD,size_t> info = getInfo() ;
if( info.second == 0U )
{
exists = false ;
return std::string() ;
}
type = info.first ;
size_t buffer_size = info.second + 1U ;
char * buffer = new char [buffer_size] ;
try
{
buffer_size = get( buffer , buffer_size ) ;
}
catch(...)
{
delete [] buffer ;
throw ;
}
std::string result( buffer , buffer_size ) ;
delete [] buffer ;
return result ;
}
size_t G::RegistryValue::get( char * buffer , size_t buffer_size ) const
{
DWORD type = 0 ;
unsigned long size = buffer_size ;
LONG rc = ::RegQueryValueEx( m_hkey.imp().m_key , m_key_name.c_str() , 0 ,
&type , reinterpret_cast<unsigned char*>(buffer) , &size ) ;
if( rc != ERROR_SUCCESS )
{
std::ostringstream ss ;
ss << "get: RegQueryValueEx(" << m_key_name << "): " << rc ;
throw ValueError( ss.str() ) ;
}
return static_cast<size_t>(size) ;
}
std::pair<g_uint32_t,size_t> G::RegistryValue::getInfo() const
{
std::pair<DWORD,size_t> result( 0 , 0 ) ;
unsigned long size = 0 ;
LONG rc = ::RegQueryValueEx( m_hkey.imp().m_key , m_key_name.c_str() , 0 ,
&result.first , NULL , &size ) ;
if( rc != ERROR_SUCCESS && rc != ERROR_FILE_NOT_FOUND )
{
std::ostringstream ss ;
ss << "RegQueryValueEx(\"" << m_key_name << "\"): " << rc ;
throw ValueError( ss.str() ) ;
}
if( rc == ERROR_FILE_NOT_FOUND )
size = 0 ; // just in case
result.second = static_cast<size_t>(size) ;
return result ;
}
void G::RegistryValue::set( const char * p )
{
set( std::string(p?p:"") ) ;
}
void G::RegistryValue::set( const std::string & s )
{
DWORD type = REG_SZ ;
LONG rc = ::RegSetValueEx( m_hkey.imp().m_key , m_key_name.c_str() , 0 ,
type , reinterpret_cast<const BYTE*>(s.c_str()) ,
s.length() + 1U ) ;
if( rc != ERROR_SUCCESS )
throw ValueError( "RegSetValueEx" ) ;
}

View File

@ -31,7 +31,7 @@
namespace G
{
class Root ;
} ;
}
// Class: G::Root
// Description: A class which aquires special privileges.

View File

@ -18,16 +18,49 @@
//
// ===
//
// gevent_win32.cpp
// gsetter.h
//
#include "gdef.h"
#include "gnet.h"
#include "geventloop.h"
#include "gwinsock.h"
#ifndef G_SETTER_H
#define G_SETTER_H
GNet::EventLoop * GNet::EventLoop::create()
#include "gdef.h"
#include "gnoncopyable.h"
namespace G
{
return new Winsock ;
class Setter ;
}
// Class: G::Setter
// Description: A class to Manage a boolean flag
// while in scope.
//
class G::Setter : public G:: noncopyable
{
public:
explicit Setter( bool & b ) ;
// Constructor. Sets the flag.
~Setter() ;
// Destructor. Resets the flag.
private:
bool & m_b ;
} ;
inline
G::Setter::Setter( bool & b ) :
m_b(b)
{
m_b = true ;
}
inline
G::Setter::~Setter()
{
m_b = false ;
}
#endif

View File

@ -143,7 +143,7 @@ public:
// Sets the current state. Returns the old state.
private:
struct Transition
struct Transition // A private structure used by G::StateMachine<>.
{
State from ;
State to ;
@ -216,7 +216,7 @@ State StateMachine<T,State,Event,Arg>::apply( T & t , Event event , const Arg &
// look up in the multimap keyed on current-state + event
//
State state = m_state ;
Map::iterator p = m_map.find(event) ;
typename Map::iterator p = m_map.find(event) ;
for( ; p != m_map.end() && (*p).first == event ; ++p )
{
if( (*p).second.from == m_any || (*p).second.from == m_state )
@ -248,7 +248,7 @@ State StateMachine<T,State,Event,Arg>::apply( T & t , Event event , const Arg &
return m_any ;
}
} ; // namespace
} // namespace
#endif

View File

@ -30,7 +30,6 @@
#include <climits>
#include <string>
#include <sstream>
#include <xlocale>
bool G::Str::replace( std::string &s , const std::string &from ,
const std::string &to , size_t *pos_p )
@ -214,16 +213,23 @@ unsigned short G::Str::toUShort( const std::string &s , bool limited )
return ushort_val ;
}
std::string G::Str::fromInt( int i )
{
std::ostringstream ss ;
ss << i ;
return ss.str() ;
}
std::string G::Str::fromUInt( unsigned int n )
{
std::stringstream ss ;
std::ostringstream ss ;
ss << n ;
return ss.str() ;
}
std::string G::Str::fromULong( unsigned long ul )
{
std::stringstream ss ;
std::ostringstream ss ;
ss << ul ;
return ss.str() ;
}
@ -236,6 +242,13 @@ void G::Str::toLower( std::string &s )
}
}
std::string G::Str::lower( const std::string &s )
{
std::string result = s ;
toLower( result ) ;
return result ;
}
void G::Str::toUpper( std::string &s )
{
for( std::string::iterator p = s.begin() ; p != s.end() ; ++p )
@ -244,6 +257,13 @@ void G::Str::toUpper( std::string &s )
}
}
std::string G::Str::upper( const std::string &s )
{
std::string result = s ;
toUpper( result ) ;
return result ;
}
std::string G::Str::toPrintableAscii( char c , char escape )
{
if( c == escape )
@ -334,7 +354,7 @@ std::string G::Str::wrap( std::string text , const std::string & prefix_1 ,
const std::string & prefix_2 , size_t width )
{
std::string ws( " \t\n" ) ;
std::stringstream ss ;
std::ostringstream ss ;
for( bool first_line = true ; text.length() ; first_line = false )
{
const size_t prefix_length =

View File

@ -119,6 +119,9 @@ public:
// Exception: Overflow
// Exception: InvalidFormat
static std::string fromInt( int i ) ;
// Converts int 'i' to a string.
static std::string fromUInt( unsigned int ) ;
// Converts from unsigned int to a decimal string.
@ -133,6 +136,14 @@ public:
// Replaces all uppercase characters in string 's' by
// lowercase characters.
static std::string upper( const std::string &s ) ;
// Returns a copy of 's' in which all lowercase characters
// have been replaced by uppercase characters.
static std::string lower( const std::string &s ) ;
// Returns a copy of 's' in which all uppercase characters
// have been replaced by lowercase characters.
static std::string toPrintableAscii( char c , char escape = '\\' ) ;
// Returns a printable, 7-bit-ascii string representing the given
// character. Typical return values include "\\", "\n",

View File

@ -36,7 +36,7 @@ namespace G
//
typedef std::list<std::string GAllocator(std::string) > Strings ;
} ;
}
#endif

View File

@ -83,7 +83,7 @@ unsigned int G::Time::seconds() const
std::string G::Time::hhmmss( const char * sep )
{
if( sep == NULL ) sep = "" ;
std::stringstream ss ;
std::ostringstream ss ;
ss << (m_hh/10U) << (m_hh%10U) << sep << (m_mm/10U) << (m_mm%10U) << sep << (m_ss/10U) << (m_ss%10U) ;
return ss.str() ;
}
@ -91,7 +91,7 @@ std::string G::Time::hhmmss( const char * sep )
std::string G::Time::hhmm( const char * sep )
{
if( sep == NULL ) sep = "" ;
std::stringstream ss ;
std::ostringstream ss ;
ss << (m_hh/10U) << (m_hh%10U) << sep << (m_mm/10U) << (m_mm%10U) ;
return ss.str() ;
}

View File

@ -32,7 +32,7 @@
namespace G
{
class Time ;
} ;
}
// Class: G::Time
// Description: A simple time-of-day (hh/mm/ss) class.

View File

@ -17,15 +17,15 @@
# Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
#
EXTRA_DIST=gclient_win32.cpp \
EXTRA_DIST=\
gclient_win32.cpp \
gdescriptor_win32.cpp \
gevent_win32.cpp \
geventloop_win32.cpp \
glocal_win32.cpp \
grequest.cpp \
gresolve_win32.cpp \
gsocket_win32.cpp \
gwinsock.cpp
INCLUDES = -I$(top_srcdir)/lib/gcc2.95 -I$(top_srcdir)/src/glib
gsocket_win32.cpp
INCLUDES = -I$(top_srcdir)/lib/$(COMPILER_VERSION) -I$(top_srcdir)/src/glib
noinst_LIBRARIES = libgnet.a
libgnet_a_SOURCES = gaddress_ipv4.cpp \
gclient.cpp \
@ -33,8 +33,8 @@ libgnet_a_SOURCES = gaddress_ipv4.cpp \
gconnection.cpp \
gdescriptor_unix.cpp \
geventloop.cpp \
geventloop_unix.cpp \
geventhandler.cpp \
gevent_unix.cpp \
geventserver.cpp \
glinebuffer.cpp \
glocal_unix.cpp \
@ -43,7 +43,6 @@ libgnet_a_SOURCES = gaddress_ipv4.cpp \
gresolve.cpp \
gresolve_ipv4.cpp \
gresolve_unix.cpp \
gselect.cpp \
gserver.cpp \
gsocket.cpp \
gsocket_unix.cpp \
@ -62,8 +61,6 @@ libgnet_a_SOURCES = gaddress_ipv4.cpp \
gnet.h \
grequest.h \
gresolve.h \
gselect.h \
gserver.h \
gsocket.h \
gtimer.h \
gwinsock.h
gtimer.h

View File

@ -79,6 +79,7 @@ POST_UNINSTALL = :
AR = @AR@
AWK = @AWK@
CC = @CC@
COMPILER_VERSION = @COMPILER_VERSION@
CXX = @CXX@
GZIP = @GZIP@
HAVE_DOXYGEN = @HAVE_DOXYGEN@
@ -96,11 +97,11 @@ e_man1dir = @e_man1dir@
e_sbindir = @e_sbindir@
e_spooldir = @e_spooldir@
EXTRA_DIST = gclient_win32.cpp gdescriptor_win32.cpp gevent_win32.cpp glocal_win32.cpp grequest.cpp gresolve_win32.cpp gsocket_win32.cpp gwinsock.cpp
EXTRA_DIST = gclient_win32.cpp gdescriptor_win32.cpp geventloop_win32.cpp glocal_win32.cpp grequest.cpp gresolve_win32.cpp gsocket_win32.cpp
INCLUDES = -I$(top_srcdir)/lib/gcc2.95 -I$(top_srcdir)/src/glib
INCLUDES = -I$(top_srcdir)/lib/$(COMPILER_VERSION) -I$(top_srcdir)/src/glib
noinst_LIBRARIES = libgnet.a
libgnet_a_SOURCES = gaddress_ipv4.cpp gclient.cpp gclient_unix.cpp gconnection.cpp gdescriptor_unix.cpp geventloop.cpp geventhandler.cpp gevent_unix.cpp geventserver.cpp glinebuffer.cpp glocal_unix.cpp glocal.cpp gmonitor.cpp gresolve.cpp gresolve_ipv4.cpp gresolve_unix.cpp gselect.cpp gserver.cpp gsocket.cpp gsocket_unix.cpp gtimer.cpp gaddress.h gclient.h gconnection.h gdescriptor.h gevent.h geventloop.h geventhandler.h geventserver.h glinebuffer.h glocal.h gmonitor.h gnet.h grequest.h gresolve.h gselect.h gserver.h gsocket.h gtimer.h gwinsock.h
libgnet_a_SOURCES = gaddress_ipv4.cpp gclient.cpp gclient_unix.cpp gconnection.cpp gdescriptor_unix.cpp geventloop.cpp geventloop_unix.cpp geventhandler.cpp geventserver.cpp glinebuffer.cpp glocal_unix.cpp glocal.cpp gmonitor.cpp gresolve.cpp gresolve_ipv4.cpp gresolve_unix.cpp gserver.cpp gsocket.cpp gsocket_unix.cpp gtimer.cpp gaddress.h gclient.h gconnection.h gdescriptor.h gevent.h geventloop.h geventhandler.h geventserver.h glinebuffer.h glocal.h gmonitor.h gnet.h grequest.h gresolve.h gserver.h gsocket.h gtimer.h
mkinstalldirs = $(SHELL) $(top_srcdir)/mkinstalldirs
CONFIG_HEADER = ../../config.h
@ -114,10 +115,10 @@ LDFLAGS = @LDFLAGS@
LIBS = @LIBS@
libgnet_a_LIBADD =
libgnet_a_OBJECTS = gaddress_ipv4.o gclient.o gclient_unix.o \
gconnection.o gdescriptor_unix.o geventloop.o geventhandler.o \
gevent_unix.o geventserver.o glinebuffer.o glocal_unix.o glocal.o \
gmonitor.o gresolve.o gresolve_ipv4.o gresolve_unix.o gselect.o \
gserver.o gsocket.o gsocket_unix.o gtimer.o
gconnection.o gdescriptor_unix.o geventloop.o geventloop_unix.o \
geventhandler.o geventserver.o glinebuffer.o glocal_unix.o glocal.o \
gmonitor.o gresolve.o gresolve_ipv4.o gresolve_unix.o gserver.o \
gsocket.o gsocket_unix.o gtimer.o
CXXFLAGS = @CXXFLAGS@
CXXCOMPILE = $(CXX) $(DEFS) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS)
CXXLD = $(CXX)
@ -228,160 +229,139 @@ distdir: $(DISTFILES)
done
gaddress_ipv4.o: gaddress_ipv4.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gnet.h \
gaddress.h ../../src/glib/gexception.h \
../../src/glib/gconvert.h ../../src/glib/gstrings.h \
../../src/glib/gstr.h ../../src/glib/gassert.h \
../../src/glib/glogoutput.h ../../src/glib/glog.h \
../../src/glib/gdebug.h
../../lib/gcc2.95/limits gnet.h gaddress.h \
../../src/glib/gexception.h ../../src/glib/gconvert.h \
../../src/glib/gstrings.h ../../src/glib/gstr.h \
../../src/glib/gassert.h ../../src/glib/glogoutput.h \
../../src/glib/glog.h ../../src/glib/gdebug.h
gclient.o: gclient.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gnet.h \
gaddress.h ../../src/glib/gexception.h gsocket.h gevent.h \
geventloop.h geventhandler.h ../../src/glib/gcredentials.h \
../../src/glib/gdatetime.h gdescriptor.h gresolve.h \
../../src/glib/groot.h ../../src/glib/gprocess.h \
../../lib/gcc2.95/limits gnet.h gaddress.h \
../../src/glib/gexception.h gsocket.h gevent.h geventloop.h \
geventhandler.h ../../src/glib/gdatetime.h gdescriptor.h \
gresolve.h ../../src/glib/groot.h ../../src/glib/gprocess.h \
../../src/glib/gpath.h ../../src/glib/gstrings.h \
../../src/glib/gnoncopyable.h gmonitor.h gclient.h \
gconnection.h gserver.h gselect.h ../../src/glib/gdebug.h \
gconnection.h gserver.h ../../src/glib/gdebug.h \
../../src/glib/glogoutput.h ../../src/glib/glog.h \
../../src/glib/gassert.h
gclient_unix.o: gclient_unix.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gnet.h \
gclient.h gaddress.h ../../src/glib/gexception.h gconnection.h \
gsocket.h gevent.h geventloop.h geventhandler.h \
../../src/glib/gcredentials.h ../../src/glib/gdatetime.h \
../../lib/gcc2.95/limits gnet.h gclient.h gaddress.h \
../../src/glib/gexception.h gconnection.h gsocket.h gevent.h \
geventloop.h geventhandler.h ../../src/glib/gdatetime.h \
gdescriptor.h
gconnection.o: gconnection.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gnet.h \
gconnection.h gaddress.h ../../src/glib/gexception.h
../../lib/gcc2.95/limits gnet.h gconnection.h gaddress.h \
../../src/glib/gexception.h
gdescriptor_unix.o: gdescriptor_unix.cpp ../../src/glib/gdef.h \
../../config.h ../../lib/gcc2.95/iostream \
../../lib/gcc2.95/sstream ../../lib/gcc2.95/xlocale \
../../lib/gcc2.95/limits gnet.h gdescriptor.h
gevent_unix.o: gevent_unix.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gnet.h \
geventloop.h geventhandler.h ../../src/glib/gcredentials.h \
../../src/glib/gdatetime.h ../../src/glib/gexception.h \
gdescriptor.h gselect.h gevent.h
../../lib/gcc2.95/sstream ../../lib/gcc2.95/limits gnet.h \
gdescriptor.h
geventhandler.o: geventhandler.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gnet.h \
geventhandler.h ../../src/glib/gcredentials.h \
../../lib/gcc2.95/limits gnet.h geventhandler.h \
../../src/glib/gdatetime.h ../../src/glib/gexception.h \
gdescriptor.h geventloop.h ../../src/glib/gdebug.h \
../../src/glib/glogoutput.h ../../src/glib/glog.h \
../../src/glib/gassert.h
geventloop.o: geventloop.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gnet.h \
geventloop.h geventhandler.h ../../src/glib/gcredentials.h \
../../lib/gcc2.95/limits gnet.h geventloop.h geventhandler.h \
../../src/glib/gdatetime.h ../../src/glib/gexception.h \
gdescriptor.h ../../src/glib/gdebug.h \
../../src/glib/glogoutput.h ../../src/glib/glog.h \
../../src/glib/gassert.h
geventloop_unix.o: geventloop_unix.cpp ../../src/glib/gdef.h \
../../config.h ../../lib/gcc2.95/iostream \
../../lib/gcc2.95/sstream ../../lib/gcc2.95/limits gnet.h \
../../src/glib/gnoncopyable.h gevent.h geventloop.h \
geventhandler.h ../../src/glib/gdatetime.h \
../../src/glib/gexception.h gdescriptor.h ../../src/glib/gstr.h \
../../src/glib/gstrings.h ../../src/glib/gsetter.h gtimer.h \
../../src/glib/gdebug.h ../../src/glib/glogoutput.h \
../../src/glib/glog.h ../../src/glib/gassert.h
geventserver.o: geventserver.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gnet.h \
geventserver.h gserver.h gsocket.h gaddress.h \
../../src/glib/gexception.h gevent.h geventloop.h \
geventhandler.h ../../src/glib/gcredentials.h \
../../src/glib/gdatetime.h gdescriptor.h gconnection.h \
gselect.h ../../src/glib/glog.h
../../lib/gcc2.95/limits gnet.h geventserver.h gserver.h \
gsocket.h gaddress.h ../../src/glib/gexception.h gevent.h \
geventloop.h geventhandler.h ../../src/glib/gdatetime.h \
gdescriptor.h gconnection.h ../../src/glib/glog.h
glinebuffer.o: glinebuffer.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gnet.h \
glinebuffer.h ../../src/glib/gstrings.h ../../src/glib/gdebug.h \
../../lib/gcc2.95/limits gnet.h glinebuffer.h \
../../src/glib/gstrings.h ../../src/glib/gdebug.h \
../../src/glib/glogoutput.h ../../src/glib/glog.h \
../../src/glib/gassert.h
glocal.o: glocal.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits glocal.h \
gnet.h gaddress.h ../../src/glib/gexception.h \
../../src/glib/gdebug.h ../../src/glib/glogoutput.h \
../../src/glib/glog.h ../../src/glib/gassert.h
../../lib/gcc2.95/limits glocal.h gnet.h gaddress.h \
../../src/glib/gexception.h ../../src/glib/gdebug.h \
../../src/glib/glogoutput.h ../../src/glib/glog.h \
../../src/glib/gassert.h
glocal_unix.o: glocal_unix.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits glocal.h \
gnet.h gaddress.h ../../src/glib/gexception.h gresolve.h \
../../src/glib/glog.h
../../lib/gcc2.95/limits glocal.h gnet.h gaddress.h \
../../src/glib/gexception.h gresolve.h ../../src/glib/glog.h
gmonitor.o: gmonitor.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gnet.h \
gmonitor.h gclient.h gaddress.h ../../src/glib/gexception.h \
gconnection.h gsocket.h gevent.h geventloop.h geventhandler.h \
../../src/glib/gcredentials.h ../../src/glib/gdatetime.h \
gdescriptor.h gserver.h gselect.h ../../src/glib/gassert.h \
../../lib/gcc2.95/limits gnet.h gmonitor.h \
../../src/glib/gnoncopyable.h gclient.h gaddress.h \
../../src/glib/gexception.h gconnection.h gsocket.h gevent.h \
geventloop.h geventhandler.h ../../src/glib/gdatetime.h \
gdescriptor.h gserver.h ../../src/glib/gassert.h \
../../src/glib/glogoutput.h ../../src/glib/glog.h
gresolve.o: gresolve.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gresolve.h \
gnet.h gaddress.h ../../src/glib/gexception.h gsocket.h \
gevent.h geventloop.h geventhandler.h \
../../src/glib/gcredentials.h ../../src/glib/gdatetime.h \
gdescriptor.h ../../src/glib/gstr.h ../../src/glib/gstrings.h \
../../lib/gcc2.95/limits gresolve.h gnet.h gaddress.h \
../../src/glib/gexception.h gsocket.h gevent.h geventloop.h \
geventhandler.h ../../src/glib/gdatetime.h gdescriptor.h \
../../src/glib/gstr.h ../../src/glib/gstrings.h \
../../src/glib/gdebug.h ../../src/glib/glogoutput.h \
../../src/glib/glog.h ../../src/glib/gassert.h
gresolve_ipv4.o: gresolve_ipv4.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gresolve.h \
gnet.h gaddress.h ../../src/glib/gexception.h
../../lib/gcc2.95/limits gresolve.h gnet.h gaddress.h \
../../src/glib/gexception.h
gresolve_unix.o: gresolve_unix.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gresolve.h \
gnet.h gaddress.h ../../src/glib/gexception.h gsocket.h \
gevent.h geventloop.h geventhandler.h \
../../src/glib/gcredentials.h ../../src/glib/gdatetime.h \
gdescriptor.h ../../src/glib/gstr.h ../../src/glib/gstrings.h \
../../lib/gcc2.95/limits gresolve.h gnet.h gaddress.h \
../../src/glib/gexception.h gsocket.h gevent.h geventloop.h \
geventhandler.h ../../src/glib/gdatetime.h gdescriptor.h \
../../src/glib/gstr.h ../../src/glib/gstrings.h \
../../src/glib/gdebug.h ../../src/glib/glogoutput.h \
../../src/glib/glog.h ../../src/glib/gassert.h
gselect.o: gselect.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gselect.h \
gnet.h gevent.h geventloop.h geventhandler.h \
../../src/glib/gcredentials.h ../../src/glib/gdatetime.h \
../../src/glib/gexception.h gdescriptor.h ../../src/glib/gstr.h \
../../src/glib/gstrings.h gtimer.h ../../src/glib/gdebug.h \
../../src/glib/glogoutput.h ../../src/glib/glog.h \
../../src/glib/gassert.h
gserver.o: gserver.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gnet.h \
gserver.h gsocket.h gaddress.h ../../src/glib/gexception.h \
gevent.h geventloop.h geventhandler.h \
../../src/glib/gcredentials.h ../../src/glib/gdatetime.h \
gdescriptor.h gconnection.h gselect.h ../../src/glib/groot.h \
../../src/glib/gprocess.h ../../src/glib/gpath.h \
../../src/glib/gstrings.h ../../src/glib/gnoncopyable.h \
gmonitor.h gclient.h ../../src/glib/gdebug.h \
../../src/glib/glogoutput.h ../../src/glib/glog.h \
../../src/glib/gassert.h ../../src/glib/gmemory.h
../../lib/gcc2.95/limits gnet.h gserver.h gsocket.h gaddress.h \
../../src/glib/gexception.h gevent.h geventloop.h \
geventhandler.h ../../src/glib/gdatetime.h gdescriptor.h \
gconnection.h ../../src/glib/groot.h ../../src/glib/gprocess.h \
../../src/glib/gpath.h ../../src/glib/gstrings.h \
../../src/glib/gnoncopyable.h gmonitor.h gclient.h \
../../src/glib/gdebug.h ../../src/glib/glogoutput.h \
../../src/glib/glog.h ../../src/glib/gassert.h \
../../src/glib/gmemory.h
gsocket.o: gsocket.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gnet.h \
../../src/glib/gassert.h ../../src/glib/glogoutput.h \
../../src/glib/glog.h gsocket.h gaddress.h \
../../src/glib/gexception.h gevent.h geventloop.h \
geventhandler.h ../../src/glib/gcredentials.h \
../../src/glib/gdatetime.h gdescriptor.h \
../../lib/gcc2.95/limits gnet.h ../../src/glib/gassert.h \
../../src/glib/glogoutput.h ../../src/glib/glog.h gsocket.h \
gaddress.h ../../src/glib/gexception.h gevent.h geventloop.h \
geventhandler.h ../../src/glib/gdatetime.h gdescriptor.h \
../../src/glib/gmemory.h ../../src/glib/gdebug.h
gsocket_unix.o: gsocket_unix.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gnet.h \
gsocket.h gaddress.h ../../src/glib/gexception.h gevent.h \
geventloop.h geventhandler.h ../../src/glib/gcredentials.h \
../../src/glib/gdatetime.h gdescriptor.h \
../../lib/gcc2.95/limits gnet.h gsocket.h gaddress.h \
../../src/glib/gexception.h gevent.h geventloop.h \
geventhandler.h ../../src/glib/gdatetime.h gdescriptor.h \
../../src/glib/gdebug.h ../../src/glib/glogoutput.h \
../../src/glib/glog.h ../../src/glib/gassert.h
gtimer.o: gtimer.cpp ../../src/glib/gdef.h ../../config.h \
../../lib/gcc2.95/iostream ../../lib/gcc2.95/sstream \
../../lib/gcc2.95/xlocale ../../lib/gcc2.95/limits gtimer.h \
gnet.h ../../src/glib/gdatetime.h ../../src/glib/gexception.h \
gevent.h geventloop.h geventhandler.h \
../../src/glib/gcredentials.h gdescriptor.h \
../../lib/gcc2.95/limits gtimer.h gnet.h \
../../src/glib/gdatetime.h ../../src/glib/gexception.h gevent.h \
geventloop.h geventhandler.h gdescriptor.h \
../../src/glib/gdebug.h ../../src/glib/glogoutput.h \
../../src/glib/glog.h ../../src/glib/gassert.h

View File

@ -34,7 +34,7 @@ namespace GNet
{
class Address ;
class AddressImp ;
} ;
}
// Class: GNet::Address
//
@ -100,6 +100,12 @@ public:
//
// See also validString().
Address( const std::string & host_or_ip , unsigned int port ) ;
// Constructor taking a host-name/ip-address and
// a port number.
//
// Throws an exception if an invalid string.
~Address() ;
// Destructor.

View File

@ -43,6 +43,7 @@ public:
explicit AddressImp( unsigned int port ) ; // (not in_port_t -- see validPort(), setPort() etc)
explicit AddressImp( const servent & s ) ;
explicit AddressImp( const std::string & s ) ;
AddressImp( const std::string & s , unsigned int port ) ;
AddressImp( unsigned int port , Address::Localhost ) ;
AddressImp( const hostent & h , unsigned int port ) ;
AddressImp( const hostent & h , const servent & s ) ;
@ -79,6 +80,8 @@ private:
static char m_port_separator ;
} ;
// ===
char GNet::AddressImp::m_port_separator = ':' ;
void GNet::AddressImp::init()
@ -141,12 +144,24 @@ GNet::AddressImp::AddressImp( const AddressImp & other )
m_inet = other.m_inet ;
}
GNet::AddressImp::AddressImp( const std::string & s , unsigned int port )
{
init() ;
std::string reason ;
if( ! setAddress( s + m_port_separator + "0" , reason ) )
throw Address::BadString( reason + ": " + s ) ;
setPort( port ) ;
}
GNet::AddressImp::AddressImp( const std::string & s )
{
init() ;
std::string reason ;
if( ! setAddress( s , reason ) )
throw Address::BadString( s ) ;
throw Address::BadString( reason + ": " + s ) ;
}
bool GNet::AddressImp::setAddress( const std::string & display_string , std::string & reason )
@ -187,7 +202,7 @@ void GNet::AddressImp::setHost( const hostent & h )
std::string GNet::AddressImp::displayString() const
{
std::stringstream ss ;
std::ostringstream ss ;
ss << hostString() ;
ss << m_port_separator << port() ;
return ss.str() ;
@ -195,7 +210,7 @@ std::string GNet::AddressImp::displayString() const
std::string GNet::AddressImp::hostString() const
{
std::stringstream ss ;
std::ostringstream ss ;
ss << ::inet_ntoa(m_inet.sin_addr) ;
return ss.str() ;
}
@ -347,7 +362,7 @@ GNet::Address::Address( const servent & s ) :
{
}
GNet::Address::Address( const sockaddr *addr , int len ) :
GNet::Address::Address( const sockaddr * addr , int len ) :
m_imp( new AddressImp(addr,len) )
{
}
@ -362,6 +377,11 @@ GNet::Address::Address( const std::string & s ) :
{
}
GNet::Address::Address( const std::string & s , unsigned int port ) :
m_imp( new AddressImp(s,port) )
{
}
GNet::Address::~Address()
{
delete m_imp ;

View File

@ -39,12 +39,12 @@ namespace
const int c_retries = 10 ; // number of retries when using a priviledged local port number
const int c_port_start = 512 ;
const int c_port_end = 1024 ;
} ;
}
namespace GNet
{
class ClientResolver ;
} ;
}
// Class: GNet::ClientResolver
// Description: A resolver class which calls ClientImp::resolveCon() when done.

View File

@ -36,7 +36,7 @@ namespace GNet
{
class Client ;
class ClientImp ;
} ;
}
// Class: GNet::Client
// Description: An application-level class for making an outgoing connection

View File

@ -31,7 +31,7 @@
namespace GNet
{
class Connection ;
} ;
}
// Class: GNet::Connection
// Description: An interface which provides address information

View File

@ -36,7 +36,7 @@ namespace GNet
Descriptor Descriptor__invalid() ;
// Returns an invalid network descriptor.
} ;
}
#endif

View File

@ -81,7 +81,7 @@ std::string GNet::EventHandlerList::asString() const
std::string GNet::EventHandlerList::asString( const EventHandlerListImp & list ) const
{
std::stringstream ss ;
std::ostringstream ss ;
const char * sep = "" ;
for( List::const_iterator p = list.begin() ; p != list.end() ; ++p )
{

View File

@ -26,7 +26,6 @@
#include "gdef.h"
#include "gnet.h"
#include "gcredentials.h"
#include "gdatetime.h"
#include "gdescriptor.h"
#include <list>
@ -38,7 +37,7 @@ namespace GNet
class EventHandlerListItem ;
class EventHandlerList ;
class TimerList ;
} ;
}
// Class: GNet::EventHandler
// Description: A pseudo-abstract base class for classes
@ -99,7 +98,7 @@ namespace GNet
{
typedef std::list< EventHandlerListItem GAllocator(EventHandlerListItem) >
EventHandlerListImp ;
} ;
}
// Class: GNet::EventHandlerList
// Description: A class which can be used in the implemention

Some files were not shown because too many files have changed in this diff Show More