This commit is contained in:
Graeme Walker 2022-04-10 12:00:00 +00:00
parent 27c01949fa
commit 2538008bc2
192 changed files with 1834 additions and 2006 deletions

View File

@ -21,3 +21,8 @@ websites for further details.
This product includes software developed by the OpenSSL Project This product includes software developed by the OpenSSL Project
for use in the OpenSSL Toolkit (http://www.openssl.org/). for use in the OpenSSL Toolkit (http://www.openssl.org/).
Translations
------------
Thanks to Per-Arne Christensen for the Norwegian translation of
the GUI.

View File

@ -1,11 +1,12 @@
E-MailRelay Change Log E-MailRelay Change Log
====================== ======================
2.2 -> 2.2.1 2.2 -> 2.3
------------ ----------
* Unix domain sockets supported (eg. "--interface=/tmp/smtp.s"). * Unix domain sockets supported (eg. "--interface=/tmp/smtp.s").
* Windows event log not used for verbose logging (prefer "--log-file"). * Windows event log not used for verbose logging (prefer "--log-file").
* New admin 'forward' command to trigger forwarding without waiting. * New admin 'forward' command to trigger forwarding without waiting.
* Optional base64 encoding of passwords in secrets files ("plain:b").
* Support for MbedTLS version 3. * Support for MbedTLS version 3.
2.1 -> 2.2 2.1 -> 2.2

View File

@ -22,7 +22,7 @@
# Additional pseudo-targets: # Additional pseudo-targets:
# * rpm - builds an rpm package using rpmbuild # * rpm - builds an rpm package using rpmbuild
# * deb - builds a deb package using debhelper # * deb - builds a deb package using debhelper
# * cmake - generates simplistic cmake files under ./build/ # * cmake - generates cmake files under ./build/
# * tidy - runs cmake-tidy # * tidy - runs cmake-tidy
# * format - runs cmake-format # * format - runs cmake-format
# #
@ -41,7 +41,7 @@
# $ sudo make deb # $ sudo make deb
# #
# and possibly: # and possibly:
# $ make cmake ; cd build ; make # $ make cmake ; make -C build
# $ make tidy TIDY=clang-tidy-10 # $ make tidy TIDY=clang-tidy-10
# $ make format FORMAT=clang-format-10 # $ make format FORMAT=clang-format-10
# $ make distcheck DISTCHECK_CONFIGURE_FLAGS=--disable-testing # $ make distcheck DISTCHECK_CONFIGURE_FLAGS=--disable-testing
@ -126,6 +126,7 @@ format:
cmake: cmake:
@chmod +x bin/make2cmake || true @chmod +x bin/make2cmake || true
bin/make2cmake bin/make2cmake
mkdir build || true test -d build || mkdir build
cd build && cmake -DCMAKE_MAKE_PROGRAM:FILEPATH=/usr/bin/make -DCMAKE_EXPORT_COMPILE_COMMANDS=ON .. cd build && cmake -DCMAKE_MAKE_PROGRAM:FILEPATH=/usr/bin/make -DCMAKE_EXPORT_COMPILE_COMMANDS=ON ..
@echo now run make from the '"build"' directory

View File

@ -24,7 +24,7 @@
# Additional pseudo-targets: # Additional pseudo-targets:
# * rpm - builds an rpm package using rpmbuild # * rpm - builds an rpm package using rpmbuild
# * deb - builds a deb package using debhelper # * deb - builds a deb package using debhelper
# * cmake - generates simplistic cmake files under ./build/ # * cmake - generates cmake files under ./build/
# * tidy - runs cmake-tidy # * tidy - runs cmake-tidy
# * format - runs cmake-format # * format - runs cmake-format
# #
@ -43,7 +43,7 @@
# $ sudo make deb # $ sudo make deb
# #
# and possibly: # and possibly:
# $ make cmake ; cd build ; make # $ make cmake ; make -C build
# $ make tidy TIDY=clang-tidy-10 # $ make tidy TIDY=clang-tidy-10
# $ make format FORMAT=clang-format-10 # $ make format FORMAT=clang-format-10
# $ make distcheck DISTCHECK_CONFIGURE_FLAGS=--disable-testing # $ make distcheck DISTCHECK_CONFIGURE_FLAGS=--disable-testing
@ -952,8 +952,9 @@ format:
cmake: cmake:
@chmod +x bin/make2cmake || true @chmod +x bin/make2cmake || true
bin/make2cmake bin/make2cmake
mkdir build || true test -d build || mkdir build
cd build && cmake -DCMAKE_MAKE_PROGRAM:FILEPATH=/usr/bin/make -DCMAKE_EXPORT_COMPILE_COMMANDS=ON .. cd build && cmake -DCMAKE_MAKE_PROGRAM:FILEPATH=/usr/bin/make -DCMAKE_EXPORT_COMPILE_COMMANDS=ON ..
@echo now run make from the '"build"' directory
# Tell versions [3.59,3.63) of GNU make to not export all variables. # 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. # Otherwise a system limit (for SysV at least) may be exceeded.

29
NEWS
View File

@ -1,28 +1,5 @@
News News
---- ----
E-MailRelay 2.2 is now fully C++11, so older compilers will not work unless Version 2.3 is a relatively minor release. The main functional change is to
they have a "-std=c++11" option or similar, and this also means that support unix domain sockets. Non-functional code changes include better
"uclibc++" is no longer supported. separation of interface and implementation in the SMTP message store.
The behaviour with respect to the "--remote-clients" command-line option is
changed in this release: previously IPv4 connections were allowed only from
the host's local address, as determined by a DNS lookup, unless using
"--remote-clients". The new implementation allows connections from any
loopback or 'private use' address, defined in RFC-1918 and RFC-5735. This
brings the IPv4 behaviour in line with IPv6, and it still honours the intent
of the "--remote-clients" option in protecting the naive user against
accidental exposore to the public internet.
Some internationalisation support has been added, using gettext() for the
main program and Qt's tr() for the GUI. See "doc/developer.txt".
A systemd unit file has been added, although by default it is only installed
as an "examples" file. Use "e_systemddir=/usr/lib/systemd/system" on the
"./configure" command-line to have it installed by "make install".
This release has a new "make tidy" option that runs "clang-tidy" over the
code and also "make cmake" to generate simple cmake files for a unix build.
Version 2.2.1 adds support for unix-domain sockets.
Thanks to Per-Arne Christensen for the Norwegian translation of the GUI.

View File

@ -1 +1 @@
2.2.1 2.3

View File

@ -298,7 +298,7 @@ sub simplepath
for my $x ( @split ) for my $x ( @split )
{ {
next if( $x eq "" || $x eq "." ) ; next if( $x eq "" || $x eq "." ) ;
if( $x eq ".." && scalar(@out) && @out[-1] ne ".." ) if( $x eq ".." && scalar(@out) && $out[-1] ne ".." )
{ {
pop @out ; pop @out ;
} }

0
bin/doxygen.sh Normal file → Executable file
View File

View File

@ -30,7 +30,7 @@ use strict ;
use FileHandle ; use FileHandle ;
$SIG{__DIE__} = sub { (my $e = join(" ",@_)) =~ s/\n/ /g ; print "<<error: $e>>\n" ; exit 99 } ; $SIG{__DIE__} = sub { (my $e = join(" ",@_)) =~ s/\n/ /g ; print "<<error: $e>>\n" ; exit 99 } ;
my $content = @ARGV[0] or die "usage error\n" ; my $content = $ARGV[0] or die "usage error\n" ;
my $verbose = 1 ; my $verbose = 1 ;
# read the bcc list from the content file # read the bcc list from the content file

View File

@ -20,7 +20,6 @@
# setup. # setup.
# #
store="__SPOOL_DIR__" store="__SPOOL_DIR__"
postmaster="root" postmaster="root"
procmail="procmail" procmail="procmail"

View File

@ -31,7 +31,7 @@ while(<$fh_in>)
if( $in_header && ( $line =~ m/^\s/ ) && scalar(@headers) ) # folding if( $in_header && ( $line =~ m/^\s/ ) && scalar(@headers) ) # folding
{ {
@headers[-1] .= "\r\n$line" ; $headers[-1] .= "\r\n$line" ;
} }
elsif( $in_header && ( $line =~ m/^$/ ) ) elsif( $in_header && ( $line =~ m/^$/ ) )
{ {

View File

@ -25,7 +25,7 @@ my $new_from = 'noreply@example.com' ;
my $new_sender = '' ; my $new_sender = '' ;
my $new_reply_to = $new_from ; my $new_reply_to = $new_from ;
my $content = @ARGV[0] or die "usage error\n" ; my $content = $ARGV[0] or die "usage error\n" ;
my $in = new FileHandle( $content , "r" ) or die ; my $in = new FileHandle( $content , "r" ) or die ;
my $out = new FileHandle( "$content.tmp" , "w" ) or die ; my $out = new FileHandle( "$content.tmp" , "w" ) or die ;

View File

@ -177,9 +177,9 @@ sub fixup
} }
# add some more whitespace # add some more whitespace
$line =~ s:(\S);$:\1 ;: ; $line =~ s:(\S);$:$1 ;: ;
$line =~ s:(\S); //(.*):\1 ; //\2: ; $line =~ s:(\S); //(.*):$1 ; //$2: ;
$line =~ s:(\S),:\1 ,:g unless ( $line =~ m/["']/ or $line =~ m://.*,: ) ; $line =~ s:(\S),:$1 ,:g unless ( $line =~ m/["']/ or $line =~ m://.*,: ) ;
print $fh_out $line , "\n" or die ; print $fh_out $line , "\n" or die ;
} }

View File

@ -353,7 +353,8 @@ sub create_touchfile
sub read_makefiles sub read_makefiles
{ {
my ( $switches , $vars ) = @_ ; my ( $switches , $vars ) = @_ ;
return AutoMakeParser::readall( "." , $switches , $vars , 1 ) ; my $verbose = 1 ;
return AutoMakeParser::readall( "." , $switches , $vars , $verbose ) ;
} }
sub cache_value sub cache_value

0
bootstrap Normal file → Executable file
View File

74
configure vendored
View File

@ -1,6 +1,6 @@
#! /bin/sh #! /bin/sh
# Guess values for system-dependent variables and create Makefiles. # Guess values for system-dependent variables and create Makefiles.
# Generated by GNU Autoconf 2.69 for E-MailRelay 2.2.1. # Generated by GNU Autoconf 2.69 for E-MailRelay 2.3.
# #
# #
# Copyright (C) 1992-1996, 1998-2012 Free Software Foundation, Inc. # Copyright (C) 1992-1996, 1998-2012 Free Software Foundation, Inc.
@ -577,8 +577,8 @@ MAKEFLAGS=
# Identity of this package. # Identity of this package.
PACKAGE_NAME='E-MailRelay' PACKAGE_NAME='E-MailRelay'
PACKAGE_TARNAME='emailrelay' PACKAGE_TARNAME='emailrelay'
PACKAGE_VERSION='2.2.1' PACKAGE_VERSION='2.3'
PACKAGE_STRING='E-MailRelay 2.2.1' PACKAGE_STRING='E-MailRelay 2.3'
PACKAGE_BUGREPORT='' PACKAGE_BUGREPORT=''
PACKAGE_URL='' PACKAGE_URL=''
@ -657,8 +657,6 @@ GCONFIG_TESTING_FALSE
GCONFIG_TESTING_TRUE GCONFIG_TESTING_TRUE
GCONFIG_MAC_FALSE GCONFIG_MAC_FALSE
GCONFIG_MAC_TRUE GCONFIG_MAC_TRUE
GCONFIG_IPV6_FALSE
GCONFIG_IPV6_TRUE
GCONFIG_INTERFACE_NAMES_FALSE GCONFIG_INTERFACE_NAMES_FALSE
GCONFIG_INTERFACE_NAMES_TRUE GCONFIG_INTERFACE_NAMES_TRUE
GCONFIG_INSTALL_HOOK_FALSE GCONFIG_INSTALL_HOOK_FALSE
@ -796,7 +794,6 @@ enable_epoll
enable_gui enable_gui
enable_install_hook enable_install_hook
enable_interface_names enable_interface_names
enable_ipv6
enable_mac enable_mac
enable_std_thread enable_std_thread
enable_testing enable_testing
@ -1378,7 +1375,7 @@ if test "$ac_init_help" = "long"; then
# Omit some internal or obsolete options to make the list less imposing. # Omit some internal or obsolete options to make the list less imposing.
# This message is too long to be a string in the A/UX 3.1 sh. # This message is too long to be a string in the A/UX 3.1 sh.
cat <<_ACEOF cat <<_ACEOF
\`configure' configures E-MailRelay 2.2.1 to adapt to many kinds of systems. \`configure' configures E-MailRelay 2.3 to adapt to many kinds of systems.
Usage: $0 [OPTION]... [VAR=VALUE]... Usage: $0 [OPTION]... [VAR=VALUE]...
@ -1445,7 +1442,7 @@ fi
if test -n "$ac_init_help"; then if test -n "$ac_init_help"; then
case $ac_init_help in case $ac_init_help in
short | recursive ) echo "Configuration of E-MailRelay 2.2.1:";; short | recursive ) echo "Configuration of E-MailRelay 2.3:";;
esac esac
cat <<\_ACEOF cat <<\_ACEOF
@ -1473,7 +1470,6 @@ Optional Features:
--enable-interface-names --enable-interface-names
allow network interface names for defining listening allow network interface names for defining listening
addresses (default yes) addresses (default yes)
--enable-ipv6 enable ipv6 (default auto)
--enable-mac enable building for mac os x (default auto) --enable-mac enable building for mac os x (default auto)
--enable-std-thread use std::thread or not (default auto) --enable-std-thread use std::thread or not (default auto)
--enable-testing enable make check tests (default yes) --enable-testing enable make check tests (default yes)
@ -1579,7 +1575,7 @@ fi
test -n "$ac_init_help" && exit $ac_status test -n "$ac_init_help" && exit $ac_status
if $ac_init_version; then if $ac_init_version; then
cat <<\_ACEOF cat <<\_ACEOF
E-MailRelay configure 2.2.1 E-MailRelay configure 2.3
generated by GNU Autoconf 2.69 generated by GNU Autoconf 2.69
Copyright (C) 2012 Free Software Foundation, Inc. Copyright (C) 2012 Free Software Foundation, Inc.
@ -2015,7 +2011,7 @@ cat >config.log <<_ACEOF
This file contains any messages produced by compilers while This file contains any messages produced by compilers while
running configure, to aid debugging if configure makes a mistake. running configure, to aid debugging if configure makes a mistake.
It was created by E-MailRelay $as_me 2.2.1, which was It was created by E-MailRelay $as_me 2.3, which was
generated by GNU Autoconf 2.69. Invocation command line was generated by GNU Autoconf 2.69. Invocation command line was
$ $0 $@ $ $0 $@
@ -2880,7 +2876,7 @@ fi
# Define the identity of the package. # Define the identity of the package.
PACKAGE='emailrelay' PACKAGE='emailrelay'
VERSION='2.2.1' VERSION='2.3'
# Some tools Automake needs. # Some tools Automake needs.
@ -9368,52 +9364,6 @@ else
fi fi
# Check whether --enable-ipv6 was given.
if test "${enable_ipv6+set}" = set; then :
enableval=$enable_ipv6;
fi
if test "$enable_ipv6" = "no"
then
gconfig_use_ipv6="no"
else
if test "$gconfig_cv_ipv6" = "no"
then
if test "$enable_ipv6" = "yes"
then
{ $as_echo "$as_me:${as_lineno-$LINENO}: WARNING: ignoring --enable-ipv6" >&5
$as_echo "$as_me: WARNING: ignoring --enable-ipv6" >&2;}
fi
gconfig_use_ipv6="no"
else
gconfig_use_ipv6="yes"
fi
fi
if test "$enable_ipv6" != "no" -a "$gconfig_use_ipv6" = "no"
then
gconfig_warnings="$gconfig_warnings ipv6_ipv6_networking"
fi
if test "$gconfig_use_ipv6" = "yes" ; then
$as_echo "#define GCONFIG_ENABLE_IPV6 1" >>confdefs.h
else
$as_echo "#define GCONFIG_ENABLE_IPV6 0" >>confdefs.h
fi
if test "$gconfig_use_ipv6" = "yes"; then
GCONFIG_IPV6_TRUE=
GCONFIG_IPV6_FALSE='#'
else
GCONFIG_IPV6_TRUE='#'
GCONFIG_IPV6_FALSE=
fi
# Check whether --enable-mac was given. # Check whether --enable-mac was given.
if test "${enable_mac+set}" = set; then : if test "${enable_mac+set}" = set; then :
enableval=$enable_mac; enableval=$enable_mac;
@ -10337,10 +10287,6 @@ if test -z "${GCONFIG_INTERFACE_NAMES_TRUE}" && test -z "${GCONFIG_INTERFACE_NAM
as_fn_error $? "conditional \"GCONFIG_INTERFACE_NAMES\" was never defined. as_fn_error $? "conditional \"GCONFIG_INTERFACE_NAMES\" was never defined.
Usually this means the macro was only invoked conditionally." "$LINENO" 5 Usually this means the macro was only invoked conditionally." "$LINENO" 5
fi fi
if test -z "${GCONFIG_IPV6_TRUE}" && test -z "${GCONFIG_IPV6_FALSE}"; then
as_fn_error $? "conditional \"GCONFIG_IPV6\" was never defined.
Usually this means the macro was only invoked conditionally." "$LINENO" 5
fi
if test -z "${GCONFIG_MAC_TRUE}" && test -z "${GCONFIG_MAC_FALSE}"; then if test -z "${GCONFIG_MAC_TRUE}" && test -z "${GCONFIG_MAC_FALSE}"; then
as_fn_error $? "conditional \"GCONFIG_MAC\" was never defined. as_fn_error $? "conditional \"GCONFIG_MAC\" was never defined.
Usually this means the macro was only invoked conditionally." "$LINENO" 5 Usually this means the macro was only invoked conditionally." "$LINENO" 5
@ -10782,7 +10728,7 @@ cat >>$CONFIG_STATUS <<\_ACEOF || ac_write_fail=1
# report actual input values of CONFIG_FILES etc. instead of their # report actual input values of CONFIG_FILES etc. instead of their
# values after options handling. # values after options handling.
ac_log=" ac_log="
This file was extended by E-MailRelay $as_me 2.2.1, which was This file was extended by E-MailRelay $as_me 2.3, which was
generated by GNU Autoconf 2.69. Invocation command line was generated by GNU Autoconf 2.69. Invocation command line was
CONFIG_FILES = $CONFIG_FILES CONFIG_FILES = $CONFIG_FILES
@ -10848,7 +10794,7 @@ _ACEOF
cat >>$CONFIG_STATUS <<_ACEOF || ac_write_fail=1 cat >>$CONFIG_STATUS <<_ACEOF || ac_write_fail=1
ac_cs_config="`$as_echo "$ac_configure_args" | sed 's/^ //; s/[\\""\`\$]/\\\\&/g'`" ac_cs_config="`$as_echo "$ac_configure_args" | sed 's/^ //; s/[\\""\`\$]/\\\\&/g'`"
ac_cs_version="\\ ac_cs_version="\\
E-MailRelay config.status 2.2.1 E-MailRelay config.status 2.3
configured by $0, generated by GNU Autoconf 2.69, configured by $0, generated by GNU Autoconf 2.69,
with options \\"\$ac_cs_config\\" with options \\"\$ac_cs_config\\"

View File

@ -19,7 +19,7 @@ dnl
dnl Process this file with autoconf to produce a configure script. dnl Process this file with autoconf to produce a configure script.
dnl dnl
AC_INIT([E-MailRelay],[2.2.1],[],[emailrelay]) AC_INIT([E-MailRelay],[2.3],[],[emailrelay])
AC_CONFIG_SRCDIR([src/glib/gdef.h]) AC_CONFIG_SRCDIR([src/glib/gdef.h])
AC_CONFIG_MACRO_DIR([m4]) AC_CONFIG_MACRO_DIR([m4])
AM_INIT_AUTOMAKE([no-define]) AM_INIT_AUTOMAKE([no-define])
@ -90,8 +90,6 @@ AC_ARG_ENABLE([install-hook],AS_HELP_STRING([--enable-install-hook],[enable fixi
GCONFIG_FN_ENABLE_INSTALL_HOOK GCONFIG_FN_ENABLE_INSTALL_HOOK
AC_ARG_ENABLE([interface-names],AS_HELP_STRING([--enable-interface-names],[allow network interface names for defining listening addresses (default yes)])) AC_ARG_ENABLE([interface-names],AS_HELP_STRING([--enable-interface-names],[allow network interface names for defining listening addresses (default yes)]))
GCONFIG_FN_ENABLE_INTERFACE_NAMES GCONFIG_FN_ENABLE_INTERFACE_NAMES
AC_ARG_ENABLE([ipv6],AS_HELP_STRING([--enable-ipv6],[enable ipv6 (default auto)]))
GCONFIG_FN_ENABLE_IPV6
AC_ARG_ENABLE([mac],AS_HELP_STRING([--enable-mac],[enable building for mac os x (default auto)])) AC_ARG_ENABLE([mac],AS_HELP_STRING([--enable-mac],[enable building for mac os x (default auto)]))
GCONFIG_FN_ENABLE_MAC GCONFIG_FN_ENABLE_MAC
AC_ARG_ENABLE([std-thread],AS_HELP_STRING([--enable-std-thread],[use std::thread or not (default auto)])) AC_ARG_ENABLE([std-thread],AS_HELP_STRING([--enable-std-thread],[use std::thread or not (default auto)]))

View File

@ -117,10 +117,10 @@ then
fi fi
if test -d "$MBEDTLS_DIR" if test -d "$MBEDTLS_DIR"
then then
echo "configure.sh: mbedtls directory exists: adding --with-mbedtls and CPPFLAGS=-I$MBEDTLS_DIR/include etc" echo "configure.sh: mbedtls directory exists: adding --with-mbedtls and CXXFLAGS=-I$MBEDTLS_DIR/include etc"
with_mbedtls="--with-mbedtls" with_mbedtls="--with-mbedtls"
make_mbedtls=1 make_mbedtls=1
export CPPFLAGS="$CPPFLAGS -I`pwd`/$MBEDTLS_DIR/include" export CXXFLAGS="$CXXFLAGS -I`pwd`/$MBEDTLS_DIR/include"
export LDFLAGS="$LDFLAGS -L`pwd`/$MBEDTLS_DIR/library" export LDFLAGS="$LDFLAGS -L`pwd`/$MBEDTLS_DIR/library"
fi fi
@ -139,7 +139,7 @@ then
export GCONFIG_WINDMC="$TARGET-windmc" export GCONFIG_WINDMC="$TARGET-windmc"
export GCONFIG_WINDRES="$TARGET-windres" export GCONFIG_WINDRES="$TARGET-windres"
export CXXFLAGS="$CXXFLAGS -std=c++11 -pthread" export CXXFLAGS="$CXXFLAGS -std=c++11 -pthread"
#export CPPFLAGS="$CPPFLAGS -D_WIN32_WINNT=0x0501" eg. for Windows XP, otherwise whatever mingw defaults to #export CXXFLAGS="$CXXFLAGS -D_WIN32_WINNT=0x0501" eg. for Windows XP, otherwise whatever mingw defaults to
export LDFLAGS="$LDFLAGS -pthread" export LDFLAGS="$LDFLAGS -pthread"
if test -x "`which $CXX`" ; then : ; else echo "error: no mingw c++ compiler: [$CXX]\n" ; exit 1 ; fi if test -x "`which $CXX`" ; then : ; else echo "error: no mingw c++ compiler: [$CXX]\n" ; exit 1 ; fi
( echo msbuild . ; echo qt-x86 . ; echo qt-x64 . ; echo cmake . ; echo msvc . ) > winbuild.cfg ( echo msbuild . ; echo qt-x86 . ; echo qt-x64 . ; echo cmake . ; echo msvc . ) > winbuild.cfg
@ -184,35 +184,35 @@ then
: :
elif test "0$opt_openwrt" -eq 1 elif test "0$opt_openwrt" -eq 1
then then
TARGET="mips-openwrt-linux-musl" TARGET="mips-openwrt-linux-musl"
SDK_DIR="`find $HOME -maxdepth 3 -type d -iname openwrt-sdk\* 2>/dev/null | sort | head -1`" SDK_DIR="`find $HOME -maxdepth 3 -type d -iname openwrt-sdk\* 2>/dev/null | sort | head -1`"
SDK_TOOLCHAIN_DIR="`find \"$SDK_DIR/staging_dir\" -type d -iname toolchain-\* 2>/dev/null | sort | head -1`" SDK_TOOLCHAIN_DIR="`find \"$SDK_DIR/staging_dir\" -type d -iname toolchain-\* 2>/dev/null | sort | head -1`"
SDK_TARGET_DIR="`find \"$SDK_DIR/staging_dir\" -type d -iname target-\* 2>/dev/null | sort | head -1`" SDK_TARGET_DIR="`find \"$SDK_DIR/staging_dir\" -type d -iname target-\* 2>/dev/null | sort | head -1`"
export CC="$SDK_TOOLCHAIN_DIR/bin/$TARGET-gcc" export CC="$SDK_TOOLCHAIN_DIR/bin/$TARGET-gcc"
export CXX="$SDK_TOOLCHAIN_DIR/bin/$TARGET-c++" export CXX="$SDK_TOOLCHAIN_DIR/bin/$TARGET-c++"
export AR="$SDK_TOOLCHAIN_DIR/bin/$TARGET-ar" export AR="$SDK_TOOLCHAIN_DIR/bin/$TARGET-ar"
export STRIP="$SDK_TOOLCHAIN_DIR/bin/$TARGET-strip" export STRIP="$SDK_TOOLCHAIN_DIR/bin/$TARGET-strip"
export CXXFLAGS="-fno-rtti -fno-threadsafe-statics -Os $CXXFLAGS" export CXXFLAGS="-fno-rtti -fno-threadsafe-statics -Os $CXXFLAGS"
export CPPFLAGS="$CPPFLAGS -DG_SMALL" export CXXFLAGS="$CXXFLAGS -DG_SMALL"
export LDFLAGS="$LDFLAGS -static" export LDFLAGS="$LDFLAGS -static"
export LIBS="-lgcc_eh" export LIBS="-lgcc_eh"
if test -x "$CXX" ; then : ; else echo "error: no c++ compiler for target [$TARGET]: CXX=[$CXX]\n" ; exit 1 ; fi if test -x "$CXX" ; then : ; else echo "error: no c++ compiler for target [$TARGET]: CXX=[$CXX]\n" ; exit 1 ; fi
$thisdir/configure $enable_debug --host $TARGET \ $thisdir/configure $enable_debug --host $TARGET \
--disable-gui --without-pam --without-doxygen \ --disable-gui --without-pam --without-doxygen \
$with_mbedtls --disable-std-thread \ $with_mbedtls --disable-std-thread \
--prefix=/usr --libexecdir=/usr/lib --sysconfdir=/etc \ --prefix=/usr --libexecdir=/usr/lib --sysconfdir=/etc \
--localstatedir=/var $opt_passthrough e_initdir=/etc/init.d "$@" --localstatedir=/var $opt_passthrough e_initdir=/etc/init.d "$@"
echo : echo :
echo "build with..." echo "build with..."
#echo " export PATH=\"$SDK_TOOLCHAIN_DIR/bin:\$PATH\"" #echo " export PATH=\"$SDK_TOOLCHAIN_DIR/bin:\$PATH\""
#echo " export STAGING_DIR=\"$SDK_DIR/staging_dir\"" #echo " export STAGING_DIR=\"$SDK_DIR/staging_dir\""
test "$make_mbedtls" -eq 1 && echo " make -C $MBEDTLS_DIR/library CC=$CC AR=$AR" test "$make_mbedtls" -eq 1 && echo " make -C $MBEDTLS_DIR/library CC=$CC AR=$AR"
echo " make" echo " make"
echo " make -C src/main strip" echo " make -C src/main strip"
: :
elif test "`uname`" = "NetBSD" elif test "`uname`" = "NetBSD"
then then
export CPPFLAGS="$CPPFLAGS -I/usr/X11R7/include" export CXXFLAGS="$CXXFLAGS -I/usr/X11R7/include"
export LDFLAGS="$LDFLAGS -L/usr/X11R7/lib" export LDFLAGS="$LDFLAGS -L/usr/X11R7/lib"
$thisdir/configure $enable_debug $with_mbedtls \ $thisdir/configure $enable_debug $with_mbedtls \
--prefix=/usr --libexecdir=/usr/lib --sysconfdir=/etc \ --prefix=/usr --libexecdir=/usr/lib --sysconfdir=/etc \
@ -220,7 +220,7 @@ then
: :
elif test "`uname`" = "FreeBSD" elif test "`uname`" = "FreeBSD"
then then
export CPPFLAGS="$CPPFLAGS -I/usr/local/include -I/usr/local/include/libav" export CXXFLAGS="$CXXFLAGS -I/usr/local/include -I/usr/local/include/libav"
export LDFLAGS="$LDFLAGS -L/usr/local/lib -L/usr/local/lib/libav" export LDFLAGS="$LDFLAGS -L/usr/local/lib -L/usr/local/lib/libav"
$thisdir/configure $enable_debug $with_mbedtls \ $thisdir/configure $enable_debug $with_mbedtls \
--prefix=/usr/local --mandir=/usr/local/man \ --prefix=/usr/local --mandir=/usr/local/man \
@ -228,7 +228,7 @@ then
: :
elif test "`uname`" = "OpenBSD" elif test "`uname`" = "OpenBSD"
then then
export CPPFLAGS="$CPPFLAGS -I/usr/X11R6/include" export CXXFLAGS="$CXXFLAGS -I/usr/X11R6/include"
export LDFLAGS="$LDFLAGS -L/usr/X11R6/lib" export LDFLAGS="$LDFLAGS -L/usr/X11R6/lib"
$thisdir/configure $enable_debug $with_mbedtls \ $thisdir/configure $enable_debug $with_mbedtls \
--prefix=/usr/local --mandir=/usr/local/man \ --prefix=/usr/local --mandir=/usr/local/man \
@ -236,14 +236,14 @@ then
: :
elif test "`uname`" = "Darwin" elif test "`uname`" = "Darwin"
then then
export CPPFLAGS="$CPPFLAGS -I/opt/local/include -I/opt/X11/include" export CXXFLAGS="$CXXFLAGS -I/opt/local/include -I/opt/X11/include"
export LDFLAGS="$LDFLAGS -L/opt/local/lib -L/opt/X11/lib" export LDFLAGS="$LDFLAGS -L/opt/local/lib -L/opt/X11/lib"
$thisdir/configure $enable_debug $with_mbedtls \ $thisdir/configure $enable_debug $with_mbedtls \
--prefix=/opt/local --mandir=/opt/local/man $opt_passthrough "$@" --prefix=/opt/local --mandir=/opt/local/man $opt_passthrough "$@"
: :
elif test "`uname`" = "Linux" elif test "`uname`" = "Linux"
then then
export CPPFLAGS export CXXFLAGS
export LDFLAGS export LDFLAGS
$thisdir/configure $enable_debug $with_mbedtls \ $thisdir/configure $enable_debug $with_mbedtls \
--prefix=/usr --libexecdir=/usr/lib --sysconfdir=/etc \ --prefix=/usr --libexecdir=/usr/lib --sysconfdir=/etc \
@ -251,7 +251,7 @@ then
$opt_passthrough e_rundir=/run/emailrelay "$@" $opt_passthrough e_rundir=/run/emailrelay "$@"
: :
else else
export CPPFLAGS="$CPPFLAGS -I/usr/X11R7/include -I/usr/X11R6/include -I/usr/local/include -I/opt/local/include -I/opt/X11/include" export CXXFLAGS="$CXXFLAGS -I/usr/X11R7/include -I/usr/X11R6/include -I/usr/local/include -I/opt/local/include -I/opt/X11/include"
export LDFLAGS="$LDFLAGS -L/usr/X11R7/lib -L/usr/X11R6/lib -L/usr/local/lib -L/opt/local/lib -L/opt/X11/lib" export LDFLAGS="$LDFLAGS -L/usr/X11R7/lib -L/usr/X11R6/lib -L/usr/local/lib -L/opt/local/lib -L/opt/X11/lib"
$thisdir/configure $enable_debug $with_mbedtls $opt_passthrough "$@" $thisdir/configure $enable_debug $with_mbedtls $opt_passthrough "$@"
fi fi

5
debian/changelog vendored
View File

@ -1,7 +1,8 @@
emailrelay (2.2.1) unstable; urgency=low emailrelay (2.3) unstable; urgency=low
* Unix domain sockets supported (eg. "--interface=/tmp/smtp.s"). * Unix domain sockets supported (eg. "--interface=/tmp/smtp.s").
* Windows event log not used for verbose logging (prefer "--log-file"). * Windows event log not used for verbose logging (prefer "--log-file").
* New admin 'forward' command to trigger forwarding without waiting. * New admin 'forward' command to trigger forwarding without waiting.
* Optional base64 encoding of passwords in secrets files ("plain:b").
* Support for MbedTLS version 3. * Support for MbedTLS version 3.
-- maintainer graeme_walker <graeme_walker@users.sourceforge.net> Tue, 14 Feb 2022 00:00:00 +0000 -- maintainer graeme_walker <graeme_walker@users.sourceforge.net> Tue, 14 Feb 2022 00:00:00 +0000
@ -27,7 +28,7 @@ emailrelay (2.1) unstable; urgency=low
* New "--idle-timeout" option for server-side connections. * New "--idle-timeout" option for server-side connections.
* Support for RFC-5782 DNSBL blocking ("--dnsbl"). * Support for RFC-5782 DNSBL blocking ("--dnsbl").
* Filter scripts are given the path of the envelope file in argv2. * Filter scripts are given the path of the envelope file in argv2.
* Message files can be editied by "--client-filter" scripts. * Message files can be edited by "--client-filter" scripts.
* Better support for CRAM-SHAx authentication. * Better support for CRAM-SHAx authentication.
* New "--client-auth-config" and "--server-auth-config" options. * New "--client-auth-config" and "--server-auth-config" options.
* New "--show" option on windows to better control the user interface style. * New "--show" option on windows to better control the user interface style.

2
debian/postinst vendored
View File

@ -27,7 +27,7 @@ emailrelay_fix_html()
emailrelay_create_config() emailrelay_create_config()
{ {
if test ! -f /etc/emailrelay.conf -a -f /etc/emailrelay.conf.template if test ! -e /etc/emailrelay.conf -a -f /etc/emailrelay.conf.template
then then
cp /etc/emailrelay.conf.template /etc/emailrelay.conf cp /etc/emailrelay.conf.template /etc/emailrelay.conf
fi fi

View File

@ -9,11 +9,12 @@
<!-- index:0::::E-MailRelay Change Log --> <!-- index:0::::E-MailRelay Change Log -->
<div class="div-main"> <div class="div-main">
<h1><a class="a-header" name="H_1">E-MailRelay Change Log</a></h1> <!-- index:1:H:E-MailRelay Change Log --> <h1><a class="a-header" name="H_1">E-MailRelay Change Log</a></h1> <!-- index:1:H:E-MailRelay Change Log -->
<h2><a class="a-header" name="SH_1_1">2.2 -> 2.2.1</a></h2> <!-- index:2:SH:1:1:2.2 -> 2.2.1 --> <h2><a class="a-header" name="SH_1_1">2.2 -> 2.3</a></h2> <!-- index:2:SH:1:1:2.2 -> 2.3 -->
<ul> <ul>
<li>Unix domain sockets supported (eg. <em>--interface=/tmp/smtp.s</em>.</li> <li>Unix domain sockets supported (eg. <em>--interface=/tmp/smtp.s</em>.</li>
<li>Windows event log not used for verbose logging (prefer <em>--log-file</em>).</li> <li>Windows event log not used for verbose logging (prefer <em>--log-file</em>).</li>
<li>New admin <em>forward</em> command to trigger forwarding without waiting.</li> <li>New admin <em>forward</em> command to trigger forwarding without waiting.</li>
<li>Optional base64 encoding of passwords in secrets files (<em>plain:b</em>).</li>
<li>Support for MbedTLS version 3.</li> <li>Support for MbedTLS version 3.</li>
</ul> </ul>
<h2><a class="a-header" name="SH_1_2">2.1 -> 2.2</a></h2> <!-- index:2:SH:1:2:2.1 -> 2.2 --> <h2><a class="a-header" name="SH_1_2">2.1 -> 2.2</a></h2> <!-- index:2:SH:1:2:2.1 -> 2.2 -->

View File

@ -1,12 +1,13 @@
E-MailRelay Change Log E-MailRelay Change Log
====================== ======================
2.2 -> 2.2.1 2.2 -> 2.3
------------ ----------
* Unix domain sockets supported (eg. `--interface=/tmp/smtp.s`). * Unix domain sockets supported (eg. `--interface=/tmp/smtp.s`).
* Windows event log not used for verbose logging (prefer `--log-file`). * Windows event log not used for verbose logging (prefer `--log-file`).
* New admin `forward` command to trigger forwarding without waiting. * New admin `forward` command to trigger forwarding without waiting.
* Optional base64 encoding of passwords in secrets files (`plain:b`).
* Support for MbedTLS version 3. * Support for MbedTLS version 3.
2.1 -> 2.2 2.1 -> 2.2
@ -34,7 +35,7 @@ E-MailRelay Change Log
* New `--idle-timeout` option for server-side connections. * New `--idle-timeout` option for server-side connections.
* Support for [RFC-5782][] [DNSBL][] blocking (`--dnsbl`). * Support for [RFC-5782][] [DNSBL][] blocking (`--dnsbl`).
* Filter scripts are given the path of the envelope file in argv2. * Filter scripts are given the path of the envelope file in argv2.
* Message files can be editied by `--client-filter` scripts. * Message files can be edited by `--client-filter` scripts.
* Better support for CRAM-SHAx authentication. * Better support for CRAM-SHAx authentication.
* New `--client-auth-config` and `--server-auth-config` options. * New `--client-auth-config` and `--server-auth-config` options.
* New `--show` option on windows to better control the user interface style. * New `--show` option on windows to better control the user interface style.

View File

@ -2,12 +2,13 @@
E-MailRelay Change Log E-MailRelay Change Log
********************** **********************
2.2 -> 2.2.1 2.2 -> 2.3
============ ==========
* Unix domain sockets supported (eg. *--interface=/tmp/smtp.s*). * Unix domain sockets supported (eg. *--interface=/tmp/smtp.s*).
* Windows event log not used for verbose logging (prefer *--log-file*). * Windows event log not used for verbose logging (prefer *--log-file*).
* New admin *forward* command to trigger forwarding without waiting. * New admin *forward* command to trigger forwarding without waiting.
* Optional base64 encoding of passwords in secrets files (*plain:b*).
* Support for MbedTLS version 3. * Support for MbedTLS version 3.
2.1 -> 2.2 2.1 -> 2.2

View File

@ -1,11 +1,12 @@
E-MailRelay Change Log E-MailRelay Change Log
====================== ======================
2.2 -> 2.2.1 2.2 -> 2.3
------------ ----------
* Unix domain sockets supported (eg. "--interface=/tmp/smtp.s"). * Unix domain sockets supported (eg. "--interface=/tmp/smtp.s").
* Windows event log not used for verbose logging (prefer "--log-file"). * Windows event log not used for verbose logging (prefer "--log-file").
* New admin "forward" command to trigger forwarding without waiting. * New admin "forward" command to trigger forwarding without waiting.
* Optional base64 encoding of passwords in secrets files ("plain:b").
* Support for MbedTLS version 3. * Support for MbedTLS version 3.
2.1 -> 2.2 2.1 -> 2.2

View File

@ -7,10 +7,10 @@ templates_path = ['_templates']
source_suffix = '.rst' source_suffix = '.rst'
master_doc = 'index' master_doc = 'index'
project = u'E-MailRelay' project = u'E-MailRelay'
copyright = u'2021, Graeme Walker' copyright = u'2022, Graeme Walker'
author = u'Graeme Walker' author = u'Graeme Walker'
version = u'2.2' version = u'2.3'
release = u'2.2' release = u'2.3'
language = None language = None
today_fmt = '%Y-%m-%d' today_fmt = '%Y-%m-%d'
exclude_patterns = [] exclude_patterns = []

View File

@ -195,7 +195,7 @@ Displays help text and then exits. Use with <I>--verbose</I> for more complete o
<DT><B>-H, --hidden</B> <DT><B>-H, --hidden</B>
<DD> <DD>
Windows only. Hides the application window and disables all message boxes, overriding any <I></I><I>--show</I><I></I> option. This is useful when running as a windows service. Windows only. Hides the application window and disables all message boxes, overriding any <I>--show</I> option. This is useful when running as a windows service.
<DT><B>--idle-timeout </B><I>&lt;time&gt;</I> <DT><B>--idle-timeout </B><I>&lt;time&gt;</I>
<DD> <DD>
@ -211,7 +211,7 @@ Specifies the IP network addresses or interface names used to bind listening por
<DT><B>--localedir </B><I>&lt;dir&gt;</I> <DT><B>--localedir </B><I>&lt;dir&gt;</I>
<DD> <DD>
Specifies a locale base directory where localisation message catalogues can be found. An empty directory can be used for the built-in default. Enables localisation and specifies the locale base directory where message catalogues can be found. An empty directory can be used for the built-in default.
<DT><B>-l, --log</B> <DT><B>-l, --log</B>
<DD> <DD>
@ -331,7 +331,7 @@ Selects and configures the low-level TLS library, using a comma-separated list o
<DT><B>-u, --user </B><I>&lt;username&gt;</I> <DT><B>-u, --user </B><I>&lt;username&gt;</I>
<DD> <DD>
When started as root the program switches to an non-privileged effective user-id when idle. This option can be used to define which user-id is used. Specify <I>root</I> to disable all user-id switching. Ignored on Windows. When started as root the program switches to a non-privileged effective user-id when idle. This option can be used to define the idle user-id and also the group ownership of new files and sockets. Specify <I>root</I> to disable all user-id switching. Ignored on Windows.
<DT><B>-v, --verbose</B> <DT><B>-v, --verbose</B>
<DD> <DD>
@ -366,7 +366,8 @@ Graeme Walker, mailto:<A HREF="mailto:graeme_walker@users.sourceforge.net">graem
</DL> </DL>
<HR> <HR>
This document was created by This document was created by
<A HREF="lynxcgi:./cgi-bin/man/man2html">man2html</A>, <A HREF="lynxcgi:FOO/cgi-bin/man/man2html">man2html</A>,
using the manual pages.<BR> using the manual pages.<BR>
</BODY> </BODY>
</HTML> </HTML>
<!-- Copyright (C) 2001-2021 Graeme Walker <graeme_walker@users.sourceforge.net>. All rights reserved. -->

View File

@ -146,7 +146,7 @@ Allow forwarding to continue even if some recipient addresses on an e-mail envel
Displays help text and then exits. Use with \fI--verbose\fR for more complete output. Displays help text and then exits. Use with \fI--verbose\fR for more complete output.
.TP .TP
.B \-H, --hidden .B \-H, --hidden
Windows only. Hides the application window and disables all message boxes, overriding any \fI\fR\fI--show\fR\fI\fR option. This is useful when running as a windows service. Windows only. Hides the application window and disables all message boxes, overriding any \fI--show\fR option. This is useful when running as a windows service.
.TP .TP
.B --idle-timeout \fI<time>\fR .B --idle-timeout \fI<time>\fR
Specifies a timeout (in seconds) for receiving network traffic from remote SMTP and POP clients. The default is 1800 seconds. Specifies a timeout (in seconds) for receiving network traffic from remote SMTP and POP clients. The default is 1800 seconds.
@ -158,7 +158,7 @@ Causes mail messages to be forwarded as they are received, even before they have
Specifies the IP network addresses or interface names used to bind listening ports. By default listening ports for incoming SMTP, POP and administration connections will bind the 'any' address for IPv4 and for IPv6, ie. \fI0.0.0.0\fR and \fI::\fR. Multiple addresses can be specified by using the option more than once or by using a comma-separated list. Use a prefix of \fIsmtp=\fR, \fIpop=\fR or \fIadmin=\fR on addresses that should apply only to those types of listening port. Any link-local IPv6 addresses must include a zone name or scope id. Interface names can be used instead of addresses, in which case all the addresses associated with that interface at startup will used for listening. When an interface name is decorated with a \fI-ipv4\fR or \fI-ipv6\fR suffix only their IPv4 or IPv6 addresses will be used (eg. \fIppp0-ipv4\fR). Specifies the IP network addresses or interface names used to bind listening ports. By default listening ports for incoming SMTP, POP and administration connections will bind the 'any' address for IPv4 and for IPv6, ie. \fI0.0.0.0\fR and \fI::\fR. Multiple addresses can be specified by using the option more than once or by using a comma-separated list. Use a prefix of \fIsmtp=\fR, \fIpop=\fR or \fIadmin=\fR on addresses that should apply only to those types of listening port. Any link-local IPv6 addresses must include a zone name or scope id. Interface names can be used instead of addresses, in which case all the addresses associated with that interface at startup will used for listening. When an interface name is decorated with a \fI-ipv4\fR or \fI-ipv6\fR suffix only their IPv4 or IPv6 addresses will be used (eg. \fIppp0-ipv4\fR).
.TP .TP
.B --localedir \fI<dir>\fR .B --localedir \fI<dir>\fR
Specifies a locale base directory where localisation message catalogues can be found. An empty directory can be used for the built-in default. Enables localisation and specifies the locale base directory where message catalogues can be found. An empty directory can be used for the built-in default.
.TP .TP
.B \-l, --log .B \-l, --log
Enables logging to the standard error stream and to the syslog. The \fI--close-stderr\fR and \fI--no-syslog\fR options can be used to disable output to standard error stream and the syslog separately. Note that \fI--as-server\fR, \fI--as-client\fR and \fI--as-proxy\fR imply \fI--log\fR, and \fI--as-server\fR and \fI--as-proxy\fR also imply \fI--close-stderr\fR. Enables logging to the standard error stream and to the syslog. The \fI--close-stderr\fR and \fI--no-syslog\fR options can be used to disable output to standard error stream and the syslog separately. Note that \fI--as-server\fR, \fI--as-client\fR and \fI--as-proxy\fR imply \fI--log\fR, and \fI--as-server\fR and \fI--as-proxy\fR also imply \fI--close-stderr\fR.
@ -248,7 +248,7 @@ When used with \fI--log\fR this option enables logging to the syslog even if the
Selects and configures the low-level TLS library, using a comma-separated list of keywords. If OpenSSL and mbedTLS are both built in then keywords of \fIopenssl\fR and \fImbedtls\fR will select one or the other. Keywords like \fItlsv1.0\fR can be used to set a minimum TLS protocol version, or \fI-tlsv1.2\fR to set a maximum version. Selects and configures the low-level TLS library, using a comma-separated list of keywords. If OpenSSL and mbedTLS are both built in then keywords of \fIopenssl\fR and \fImbedtls\fR will select one or the other. Keywords like \fItlsv1.0\fR can be used to set a minimum TLS protocol version, or \fI-tlsv1.2\fR to set a maximum version.
.TP .TP
.B \-u, --user \fI<username>\fR .B \-u, --user \fI<username>\fR
When started as root the program switches to an non-privileged effective user-id when idle. This option can be used to define which user-id is used. Specify \fIroot\fR to disable all user-id switching. Ignored on Windows. When started as root the program switches to a non-privileged effective user-id when idle. This option can be used to define the idle user-id and also the group ownership of new files and sockets. Specify \fIroot\fR to disable all user-id switching. Ignored on Windows.
.TP .TP
.B \-v, --verbose .B \-v, --verbose
Enables more verbose logging when used with \fI--log\fR, and more verbose help when used with \fI--help\fR. Enables more verbose logging when used with \fI--log\fR, and more verbose help when used with \fI--help\fR.

View File

@ -1,4 +1,4 @@
<!DOCTYPE HTML PUBLIC "%-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/strict.dtd"> <!DOCTYPE html>
<html> <html>
<head> <head>
<title>E-MailRelay Reference</title> <title>E-MailRelay Reference</title>
@ -8,7 +8,7 @@
<body> <body>
<!-- index:0::::E-MailRelay Reference --> <!-- index:0::::E-MailRelay Reference -->
<div class="div-main"> <div class="div-main">
<h1><a class="a-header" name="H_1">E-MailRelay Reference</a></h1> <!-- index:1:H:1::E-MailRelay Reference --> <h1><a class="a-header" name="H_1">E-MailRelay Reference</a></h1> <!-- index:1:H:E-MailRelay Reference -->
<h2><a class="a-header" name="SH_1_1">Command line usage</a></h2> <!-- index:2:SH:1:1:Command line usage --> <h2><a class="a-header" name="SH_1_1">Command line usage</a></h2> <!-- index:2:SH:1:1:Command line usage -->
<p> <p>
The <em>emailrelay</em> program supports the following command-line usage: The <em>emailrelay</em> program supports the following command-line usage:
@ -224,7 +224,7 @@
<dt>--hidden (-H)</dt> <dt>--hidden (-H)</dt>
<dd> <dd>
Windows only. Hides the application window and disables all message boxes, Windows only. Hides the application window and disables all message boxes,
overriding any <em></em>--show<em></em> option. This is useful when running as a windows overriding any <em>--show</em> option. This is useful when running as a windows
service. service.
</dd> </dd>
<dt>--idle-timeout &lt;time&gt;</dt> <dt>--idle-timeout &lt;time&gt;</dt>
@ -255,8 +255,9 @@
</dd> </dd>
<dt>--localedir &lt;dir&gt;</dt> <dt>--localedir &lt;dir&gt;</dt>
<dd> <dd>
Specifies a locale base directory where localisation message catalogues can Enables localisation and specifies the locale base directory where message
be found. An empty directory can be used for the built-in default. catalogues can be found. An empty directory can be used for the built-in
default.
</dd> </dd>
<dt>--log (-l)</dt> <dt>--log (-l)</dt>
<dd> <dd>
@ -437,9 +438,10 @@
</dd> </dd>
<dt>--user &lt;username&gt; (-u)</dt> <dt>--user &lt;username&gt; (-u)</dt>
<dd> <dd>
When started as root the program switches to an non-privileged effective When started as root the program switches to a non-privileged effective
user-id when idle. This option can be used to define which user-id is used. user-id when idle. This option can be used to define the idle user-id and
Specify <em>root</em> to disable all user-id switching. Ignored on Windows. also the group ownership of new files and sockets. Specify <em>root</em> to
disable all user-id switching. Ignored on Windows.
</dd> </dd>
<dt>--verbose (-v)</dt> <dt>--verbose (-v)</dt>
<dd> <dd>
@ -503,7 +505,7 @@
<li>as each message is submitted, just before receipt is acknowledged (<em>--immediate</em>)</li> <li>as each message is submitted, just before receipt is acknowledged (<em>--immediate</em>)</li>
<li>as soon as the submitting client connection disconnects (<em>--forward-on-disconnect</em>)</li> <li>as soon as the submitting client connection disconnects (<em>--forward-on-disconnect</em>)</li>
<li>periodically (<em>--poll=<seconds></em>)</li> <li>periodically (<em>--poll=<seconds></em>)</li>
<li>on demand using the administration interface's <em>flush</em> command (<em>--admin=<port></em>)</li> <li>on demand using the administration interface's <em>forward</em> command (<em>--admin=<port></em>)</li>
<li>when a <em>--filter</em> script exits with an exit code of 103</li> <li>when a <em>--filter</em> script exits with an exit code of 103</li>
</ul> </ul>
@ -753,13 +755,20 @@ emailrelay --as-client=example.com:smtp --client-auth=/etc/emailrelay-server.aut
</p> </p>
<p> <p>
The first two fields are case-insensitive. The <em>xtext</em> encoding scheme is The <em>xtext</em> encoding scheme is defined properly in RFC-3461, but basically it
defined properly in RFC-3461, but basically it says that non-alphanumeric says that non-alphanumeric characters (including space, <em>+</em>, <em>#</em> and <em>=</em>) should
characters (including space, <em>+</em>, <em>#</em> and <em>=</em>) should be represented in be represented in uppercase hexadecimal ascii as <em>+XX</em>. So a space should be
uppercase hexadecimal ascii as <em>+XX</em>. So a space should be written as <em>+20</em>; written as <em>+20</em>; <em>+</em> as <em>+2B</em>; <em>#</em> as <em>+23</em>; and <em>=</em> as <em>+3D</em>.
<em>+</em> as <em>+2B</em>; <em>#</em> as <em>+23</em>; and <em>=</em> as <em>+3D</em>. Also note that modern email </p>
services will expect userids and passwords containing non-ASCII characters to
use UTF-8 encoding with RFC-4013 normalisation applied. <p>
Base64 encoding can be used instead of xtext encoding for the user identifier
and plain password by replacing <em>plain</em> by <em>plain:b</em>.
</p>
<p>
Note that modern email services will expect userids and passwords containing
non-ASCII characters to use UTF-8 encoding with RFC-4013 normalisation applied.
</p> </p>
<p> <p>
@ -775,10 +784,10 @@ emailrelay --as-client=example.com:smtp --client-auth=/etc/emailrelay-server.aut
<p> <p>
The PLAIN, LOGIN and CRAM-MD5 mechanisms can use plaintext passwords, stored The PLAIN, LOGIN and CRAM-MD5 mechanisms can use plaintext passwords, stored
in the secrets file using a password-type of <em>plain</em>. In addition, the in the secrets file using a password-type of <em>plain</em>. In addition, the
CRAM-MD5 mechanism can also use hashed passwords generated by the CRAM-MD5 mechanism can also make use of hashed passwords generated by the
<em>emailrelay-passwd</em> program and these are stored in the secrets file with a <em>emailrelay-passwd</em> program and these are stored in the secrets file with a
password-type of <em>md5</em>. (Hashed passwords are marginally more secure because password-type of <em>md5</em>. (Hashed passwords are marginally more secure because
the plaintext password which might be used on other accounts, is not easily the plaintext password which might be used on other accounts is not easily
recovered. However, hashed passwords can only be used for HMAC authentication recovered. However, hashed passwords can only be used for HMAC authentication
mechanisms that are based on the same hash function.) The XOAUTH2 mechanism mechanisms that are based on the same hash function.) The XOAUTH2 mechanism
can be used for client-side authentication using tokens that have been can be used for client-side authentication using tokens that have been
@ -872,10 +881,10 @@ server plain carol my+20password
</p> </p>
<h2><a class="a-header" name="SH_1_6">TLS encryption</a></h2> <!-- index:2:SH:1:6:TLS encryption --> <h2><a class="a-header" name="SH_1_6">TLS encryption</a></h2> <!-- index:2:SH:1:6:TLS encryption -->
<p> <p>
E-MailRelay can use negotiated TLS to encrypt SMTP and POP sessions: to enable E-MailRelay can use negotiated TLS to encrypt SMTP and POP sessions: use the
client-side TLS encryption when E-MailRelay is acting as an SMTP client use the <em>--client-tls</em> command-line option to enable client-side TLS encryption when
<em>--client-tls</em> command-line option, and to enable server-side TLS when E-MailRelay is acting as an SMTP client, and use <em>--server-tls</em> to enable
E-MailRelay is acting as an SMTP or POP server use <em>--server-tls</em>. The server-side TLS when E-MailRelay is acting as an SMTP or POP server. The
connections start off as unencrypted and the SMTP command <em>STARTTLS</em> (or the connections start off as unencrypted and the SMTP command <em>STARTTLS</em> (or the
POP <em>STLS</em> command) can be used to negotiate TLS encryption before any POP <em>STLS</em> command) can be used to negotiate TLS encryption before any
passwords are exchanged. passwords are exchanged.
@ -1083,7 +1092,7 @@ password required pam_deny.so
<pre>--as-client ipv4or6.example.com:25 --client-interface 0.0.0.0 <pre>--as-client ipv4or6.example.com:25 --client-interface 0.0.0.0
--as-client ipv4or6.example.com:25 --client-interface ::</pre> --as-client ipv4or6.example.com:25 --client-interface ::</pre>
</div><!-- div-pre --> </div><!-- div-pre -->
<h2><a class="a-header" name="SH_1_99">Unix domain sockets</a></h2> <!-- index:2:SH:1:99:Unix domain sockets --> <h2><a class="a-header" name="SH_1_9">Unix domain sockets</a></h2> <!-- index:2:SH:1:9:Unix domain sockets -->
<p> <p>
E-MailRelay on Unix will listen on unix-domain sockets instead of IPv4 or IPv6 E-MailRelay on Unix will listen on unix-domain sockets instead of IPv4 or IPv6
if the <em>--interface</em> option is given as an absolute file-system path: if the <em>--interface</em> option is given as an absolute file-system path:
@ -1093,31 +1102,37 @@ password required pam_deny.so
Eg: Eg:
</p> </p>
<div class="div-pre"> <div class="div-pre">
<pre>--interface=/run/smtp.s --port=0</pre> <pre>--interface=/run/smtp.s --port=0
</pre>
</div><!-- div-pre --> </div><!-- div-pre -->
<p> <p>
When listening on more than one unix-domain socket use the extended form of the When listening on more than one unix-domain socket use the extended form of the
"--interface" option with a prefix of "smtp=", "pop=", or "admin=": <em>--interface</em> option with a prefix of <em>smtp=</em>, <em>pop=</em>, or <em>admin=</em>:
</p> </p>
<p> <p>
Eg: Eg:
</p> </p>
<div class="div-pre"> <div class="div-pre">
<pre>--interface=smtp=/run/smtp.s --port=0 --interface=pop=/run/pop.s --pop --pop-port=0</pre> <pre>--interface=smtp=/run/smtp.s --port=0 --interface=pop=/run/pop.s --pop --pop-port=0
</pre>
</div><!-- div-pre --> </div><!-- div-pre -->
<p> <p>
The forwarding address can also be a unix-domain address: The forwarding address can also be a unix-domain address:
</p> </p>
<p> <p>
Eg: Eg:
</p> </p>
<div class="div-pre"> <div class="div-pre">
<pre>--forward-to=/run/smtp.s</pre> <pre>--forward-to=/run/smtp.s
</pre>
</div><!-- div-pre --> </div><!-- div-pre -->
<p> <p>
And it is also possible to communicate with message filters over a unix-domain And it is also possible to communicate with message filters over a unix-domain
socket: socket:
</p> </p>
<p> <p>
Eg: Eg:
</p> </p>
@ -1126,7 +1141,7 @@ password required pam_deny.so
--filter=spam:/run/spamd.s --filter=spam:/run/spamd.s
--filter=spam-edit:/run/spamd.s</pre> --filter=spam-edit:/run/spamd.s</pre>
</div><!-- div-pre --> </div><!-- div-pre -->
<h2><a class="a-header" name="SH_1_9">SOCKS</a></h2> <!-- index:2:SH:1:9:SOCKS --> <h2><a class="a-header" name="SH_1_10">SOCKS</a></h2> <!-- index:2:SH:1:10:SOCKS -->
<p> <p>
E-MailRelay can use a SOCKS 4a proxy for establishing outgoing SMTP E-MailRelay can use a SOCKS 4a proxy for establishing outgoing SMTP
connections; just append the SOCKS proxy address to the SMTP server's address, connections; just append the SOCKS proxy address to the SMTP server's address,
@ -1147,7 +1162,7 @@ password required pam_deny.so
establish the connection. The target SMTP server will see a connection coming establish the connection. The target SMTP server will see a connection coming
from the Tor exit node rather than from the E-MailRelay server. from the Tor exit node rather than from the E-MailRelay server.
</p> </p>
<h2><a class="a-header" name="SH_1_10">Address verification</a></h2> <!-- index:2:SH:1:10:Address verification --> <h2><a class="a-header" name="SH_1_11">Address verification</a></h2> <!-- index:2:SH:1:11:Address verification -->
<p> <p>
By default the E-MailRelay server will accept all recipient addresses for By default the E-MailRelay server will accept all recipient addresses for
incoming e-mails as valid. This default behaviour can be modified by using an incoming e-mails as valid. This default behaviour can be modified by using an
@ -1350,7 +1365,7 @@ catch( e )
information as returned by verifier scripts but in reverse, such as information as returned by verifier scripts but in reverse, such as
<em>0|postmaster|Local Postmaster <postmaster@eg.com></em> or <em>2|mailbox unavailable</em>. <em>0|postmaster|Local Postmaster <postmaster@eg.com></em> or <em>2|mailbox unavailable</em>.
</p> </p>
<h2><a class="a-header" name="SH_1_11">Connection blocking</a></h2> <!-- index:2:SH:1:11:Connection blocking --> <h2><a class="a-header" name="SH_1_12">Connection blocking</a></h2> <!-- index:2:SH:1:12:Connection blocking -->
<p> <p>
All incoming connections from remote network addresses are blocked by default, All incoming connections from remote network addresses are blocked by default,
but can be allowed by using the <em>--remote-clients</em>/<em>-r</em> option. This is to but can be allowed by using the <em>--remote-clients</em>/<em>-r</em> option. This is to
@ -1390,7 +1405,7 @@ catch( e )
Connections from loopback and private (RFC-1918) network addresses are never Connections from loopback and private (RFC-1918) network addresses are never
checked. checked.
</p> </p>
<h2><a class="a-header" name="SH_1_12">Security issues</a></h2> <!-- index:2:SH:1:12:Security issues --> <h2><a class="a-header" name="SH_1_13">Security issues</a></h2> <!-- index:2:SH:1:13:Security issues -->
<p> <p>
The following are some security issues that have been taken into consideration: The following are some security issues that have been taken into consideration:
</p> </p>
@ -1477,7 +1492,7 @@ catch( e )
The <em>Authentication</em>, <em>PAM Authentication</em> and <em>TLS encryption</em> sections The <em>Authentication</em>, <em>PAM Authentication</em> and <em>TLS encryption</em> sections
above also relate to security. above also relate to security.
</p> </p>
<h2><a class="a-header" name="SH_1_13">Administration interface</a></h2> <!-- index:2:SH:1:13:Administration interface --> <h2><a class="a-header" name="SH_1_14">Administration interface</a></h2> <!-- index:2:SH:1:14:Administration interface -->
<p> <p>
If enabled with the <em>--admin</em> command-line option, the E-MailRelay server will If enabled with the <em>--admin</em> command-line option, the E-MailRelay server will
provide a network interface for performing administration tasks. This is a provide a network interface for performing administration tasks. This is a
@ -1492,9 +1507,18 @@ E-MailRelay&gt; quit
</pre> </pre>
</div><!-- div-pre --> </div><!-- div-pre -->
<p> <p>
The <em>flush</em> command is used to get the E-MailRelay server to forward spooled The <em>forward</em> command is used to trigger the E-MailRelay server into forwarding
mail to the next SMTP server. The <em>forward</em> command does the same but without spooled mail to the next SMTP server.
waiting for completion. </p>
<p>
The <em>flush</em> command is similar but it uses its own connection to the SMTP
server and waits for the messages to be sent.
</p>
<p>
The <em>unfail-all</em> command can be used to remove the <em>.bad</em> filename extension
from files in the spool directory.
</p> </p>
<p> <p>
@ -1502,7 +1526,7 @@ E-MailRelay&gt; quit
network status information and activity statistics, and <em>notify</em> enables network status information and activity statistics, and <em>notify</em> enables
asynchronous event notification. asynchronous event notification.
</p> </p>
<h2><a class="a-header" name="SH_1_14">Bcc handling</a></h2> <!-- index:2:SH:1:14:Bcc handling --> <h2><a class="a-header" name="SH_1_15">Bcc handling</a></h2> <!-- index:2:SH:1:15:Bcc handling -->
<p> <p>
E-MailRelay transfers e-mail messages without changing their content in any E-MailRelay transfers e-mail messages without changing their content in any
way, other than by adding a <em>Received</em> header. In particular, if a message way, other than by adding a <em>Received</em> header. In particular, if a message
@ -1523,7 +1547,7 @@ E-MailRelay&gt; quit
An E-MailRelay <em>--filter</em> script can be used to reject messages with incorrect An E-MailRelay <em>--filter</em> script can be used to reject messages with incorrect
<em>Bcc:</em> headers, and an example script is included. <em>Bcc:</em> headers, and an example script is included.
</p> </p>
<h2><a class="a-header" name="SH_1_15">Files and directories</a></h2> <!-- index:2:SH:1:15:Files and directories --> <h2><a class="a-header" name="SH_1_16">Files and directories</a></h2> <!-- index:2:SH:1:16:Files and directories -->
<p> <p>
On Unix-like systems E-MailRelay installs by default under <em>/usr/local</em>, but On Unix-like systems E-MailRelay installs by default under <em>/usr/local</em>, but
binary distributions will probably have been built to install elsewhere. binary distributions will probably have been built to install elsewhere.
@ -1569,7 +1593,7 @@ E-MailRelay&gt; quit
It is possible to change the installation root directory after building by It is possible to change the installation root directory after building by
using <em>make DESTDIR=<root> install</em> or <em>DESTDIR=<root> make -e install</em>. using <em>make DESTDIR=<root> install</em> or <em>DESTDIR=<root> make -e install</em>.
However, this will not change the default spool directory path built into the However, this will not change the default spool directory path built into the
scripts and executables so the correct spool directory will have to be scripts and executables so the correct spool directory will then have to be
specified at run-time with the <em>--spool-dir</em> command-line option. specified at run-time with the <em>--spool-dir</em> command-line option.
</p> </p>

View File

@ -210,7 +210,7 @@ where &lt;option&gt; is:
* \-\-hidden (-H) * \-\-hidden (-H)
Windows only. Hides the application window and disables all message boxes, Windows only. Hides the application window and disables all message boxes,
overriding any `--show` option. This is useful when running as a windows overriding any `--show` option. This is useful when running as a windows
service. service.
* \-\-idle-timeout &lt;time&gt; * \-\-idle-timeout &lt;time&gt;
@ -241,8 +241,9 @@ where &lt;option&gt; is:
* \-\-localedir &lt;dir&gt; * \-\-localedir &lt;dir&gt;
Specifies a locale base directory where localisation message catalogues can Enables localisation and specifies the locale base directory where message
be found. An empty directory can be used for the built-in default. catalogues can be found. An empty directory can be used for the built-in
default.
* \-\-log (-l) * \-\-log (-l)
@ -423,9 +424,10 @@ where &lt;option&gt; is:
* \-\-user &lt;username&gt; (-u) * \-\-user &lt;username&gt; (-u)
When started as root the program switches to an non-privileged effective When started as root the program switches to a non-privileged effective
user-id when idle. This option can be used to define which user-id is used. user-id when idle. This option can be used to define the idle user-id and
Specify `root` to disable all user-id switching. Ignored on Windows. also the group ownership of new files and sockets. Specify `root` to
disable all user-id switching. Ignored on Windows.
* \-\-verbose (-v) * \-\-verbose (-v)
@ -475,7 +477,7 @@ command-line options:
* as each message is submitted, just before receipt is acknowledged (`--immediate`) * as each message is submitted, just before receipt is acknowledged (`--immediate`)
* as soon as the submitting client connection disconnects (`--forward-on-disconnect`) * as soon as the submitting client connection disconnects (`--forward-on-disconnect`)
* periodically (`--poll=<seconds>`) * periodically (`--poll=<seconds>`)
* on demand using the administration interface's `flush` command (`--admin=<port>`) * on demand using the administration interface's `forward` command (`--admin=<port>`)
* when a `--filter` script exits with an exit code of 103 * when a `--filter` script exits with an exit code of 103
These can be mixed. These can be mixed.
@ -660,13 +662,16 @@ user identifier; and the `password` field is the xtext-encoded plain password
or a base64-encoded `HMAC-MD5` state. For `client` lines the password-type can or a base64-encoded `HMAC-MD5` state. For `client` lines the password-type can
also be `oauth`. also be `oauth`.
The first two fields are case-insensitive. The `xtext` encoding scheme is The `xtext` encoding scheme is defined properly in [RFC-3461][], but basically it
defined properly in [RFC-3461][], but basically it says that non-alphanumeric says that non-alphanumeric characters (including space, `+`, `#` and `=`) should
characters (including space, `+`, `#` and `=`) should be represented in be represented in uppercase hexadecimal ascii as `+XX`. So a space should be
uppercase hexadecimal ascii as `+XX`. So a space should be written as `+20`; written as `+20`; `+` as `+2B`; `#` as `+23`; and `=` as `+3D`.
`+` as `+2B`; `#` as `+23`; and `=` as `+3D`. Also note that modern email
services will expect userids and passwords containing non-ASCII characters to Base64 encoding can be used instead of xtext encoding for the user identifier
use UTF-8 encoding with [RFC-4013][] normalisation applied. and plain password by replacing `plain` by `plain:b`.
Note that modern email services will expect userids and passwords containing
non-ASCII characters to use UTF-8 encoding with [RFC-4013][] normalisation applied.
Authentication proceeds according to an authentication 'mechanism' that is Authentication proceeds according to an authentication 'mechanism' that is
advertised by the server and selected by the client. Many authentication advertised by the server and selected by the client. Many authentication
@ -678,10 +683,10 @@ available via PAM (see below).
The PLAIN, LOGIN and CRAM-MD5 mechanisms can use plaintext passwords, stored The PLAIN, LOGIN and CRAM-MD5 mechanisms can use plaintext passwords, stored
in the secrets file using a password-type of `plain`. In addition, the in the secrets file using a password-type of `plain`. In addition, the
CRAM-MD5 mechanism can also use hashed passwords generated by the CRAM-MD5 mechanism can also make use of hashed passwords generated by the
`emailrelay-passwd` program and these are stored in the secrets file with a `emailrelay-passwd` program and these are stored in the secrets file with a
password-type of `md5`. (Hashed passwords are marginally more secure because password-type of `md5`. (Hashed passwords are marginally more secure because
the plaintext password which might be used on other accounts, is not easily the plaintext password which might be used on other accounts is not easily
recovered. However, hashed passwords can only be used for HMAC authentication recovered. However, hashed passwords can only be used for HMAC authentication
mechanisms that are based on the same hash function.) The XOAUTH2 mechanism mechanisms that are based on the same hash function.) The XOAUTH2 mechanism
can be used for client-side authentication using tokens that have been can be used for client-side authentication using tokens that have been
@ -753,10 +758,10 @@ described below.
TLS encryption TLS encryption
-------------- --------------
E-MailRelay can use negotiated TLS to encrypt SMTP and POP sessions: to enable E-MailRelay can use negotiated TLS to encrypt SMTP and POP sessions: use the
client-side TLS encryption when E-MailRelay is acting as an SMTP client use the `--client-tls` command-line option to enable client-side TLS encryption when
`--client-tls` command-line option, and to enable server-side TLS when E-MailRelay is acting as an SMTP client, and use `--server-tls` to enable
E-MailRelay is acting as an SMTP or POP server use `--server-tls`. The server-side TLS when E-MailRelay is acting as an SMTP or POP server. The
connections start off as unencrypted and the SMTP command `STARTTLS` (or the connections start off as unencrypted and the SMTP command `STARTTLS` (or the
POP `STLS` command) can be used to negotiate TLS encryption before any POP `STLS` command) can be used to negotiate TLS encryption before any
passwords are exchanged. passwords are exchanged.
@ -1169,7 +1174,7 @@ The following are some security issues that have been taken into consideration:
`root.daemon` with permissions of `-rwxrwxr-x` and messages files are created `root.daemon` with permissions of `-rwxrwxr-x` and messages files are created
with permissions of `-rw-rw----`. This allows normal users to list messages with permissions of `-rw-rw----`. This allows normal users to list messages
files but not read them. files but not read them.
The `emailrelay-submit` program is given group ownership of `daemon` with its The `emailrelay-submit` program is given group ownership of `daemon` with its
group set-user-id flag set. This allows it to create message files in the group set-user-id flag set. This allows it to create message files in the
spool directory, and the files created end up owned by the submitter but with spool directory, and the files created end up owned by the submitter but with
@ -1179,7 +1184,7 @@ The following are some security issues that have been taken into consideration:
Logging output is conditioned so that ANSI escape sequences cannot appear Logging output is conditioned so that ANSI escape sequences cannot appear
in the log. in the log.
Passwords and message content are not logged (except if using the `--debug` Passwords and message content are not logged (except if using the `--debug`
option at run time with debug logging enabled at build time). option at run time with debug logging enabled at build time).
@ -1214,9 +1219,14 @@ simple command-line interface which is compatible with `netcat` and `telnet`:
E-MailRelay> help E-MailRelay> help
E-MailRelay> quit E-MailRelay> quit
The `flush` command is used to get the E-MailRelay server to forward spooled The `forward` command is used to trigger the E-MailRelay server into forwarding
mail to the next SMTP server. The `forward` command does the same but without spooled mail to the next SMTP server.
waiting for completion.
The `flush` command is similar but it uses its own connection to the SMTP
server and waits for the messages to be sent.
The `unfail-all` command can be used to remove the `.bad` filename extension
from files in the spool directory.
The `list` command lists the messages in the spool directory, `status` provides The `list` command lists the messages in the spool directory, `status` provides
network status information and activity statistics, and `notify` enables network status information and activity statistics, and `notify` enables
@ -1274,7 +1284,7 @@ Standard ([FHS][]) use this configure command:
It is possible to change the installation root directory after building by It is possible to change the installation root directory after building by
using `make DESTDIR=<root> install` or `DESTDIR=<root> make -e install`. using `make DESTDIR=<root> install` or `DESTDIR=<root> make -e install`.
However, this will not change the default spool directory path built into the However, this will not change the default spool directory path built into the
scripts and executables so the correct spool directory will have to be scripts and executables so the correct spool directory will then have to be
specified at run-time with the `--spool-dir` command-line option. specified at run-time with the `--spool-dir` command-line option.
On Windows the installation GUI prompts for two installation directories, On Windows the installation GUI prompts for two installation directories,

View File

@ -213,7 +213,7 @@ where \<option\> is:
* --hidden (-H) * --hidden (-H)
Windows only. Hides the application window and disables all message boxes, Windows only. Hides the application window and disables all message boxes,
overriding any *--show* option. This is useful when running as a windows overriding any *--show* option. This is useful when running as a windows
service. service.
* --idle-timeout \<time\> * --idle-timeout \<time\>
@ -244,8 +244,9 @@ where \<option\> is:
* --localedir \<dir\> * --localedir \<dir\>
Specifies a locale base directory where localisation message catalogues can Enables localisation and specifies the locale base directory where message
be found. An empty directory can be used for the built-in default. catalogues can be found. An empty directory can be used for the built-in
default.
* --log (-l) * --log (-l)
@ -426,9 +427,10 @@ where \<option\> is:
* --user \<username\> (-u) * --user \<username\> (-u)
When started as root the program switches to an non-privileged effective When started as root the program switches to a non-privileged effective
user-id when idle. This option can be used to define which user-id is used. user-id when idle. This option can be used to define the idle user-id and
Specify *root* to disable all user-id switching. Ignored on Windows. also the group ownership of new files and sockets. Specify *root* to
disable all user-id switching. Ignored on Windows.
* --verbose (-v) * --verbose (-v)
@ -478,7 +480,7 @@ command-line options:
* as each message is submitted, just before receipt is acknowledged (\ *--immediate*\ ) * as each message is submitted, just before receipt is acknowledged (\ *--immediate*\ )
* as soon as the submitting client connection disconnects (\ *--forward-on-disconnect*\ ) * as soon as the submitting client connection disconnects (\ *--forward-on-disconnect*\ )
* periodically (\ *--poll=<seconds>*\ ) * periodically (\ *--poll=<seconds>*\ )
* on demand using the administration interface's *flush* command (\ *--admin=<port>*\ ) * on demand using the administration interface's *forward* command (\ *--admin=<port>*\ )
* when a *--filter* script exits with an exit code of 103 * when a *--filter* script exits with an exit code of 103
These can be mixed. These can be mixed.
@ -675,13 +677,16 @@ user identifier; and the *password* field is the xtext-encoded plain password
or a base64-encoded *HMAC-MD5* state. For *client* lines the password-type can or a base64-encoded *HMAC-MD5* state. For *client* lines the password-type can
also be *oauth*. also be *oauth*.
The first two fields are case-insensitive. The *xtext* encoding scheme is The *xtext* encoding scheme is defined properly in RFC-3461_, but basically it
defined properly in RFC-3461_, but basically it says that non-alphanumeric says that non-alphanumeric characters (including space, *+*, *#* and *=*) should
characters (including space, *+*, *#* and *=*) should be represented in be represented in uppercase hexadecimal ascii as *+XX*. So a space should be
uppercase hexadecimal ascii as *+XX*. So a space should be written as *+20*; written as *+20*; *+* as *+2B*; *#* as *+23*; and *=* as *+3D*.
*+* as *+2B*; *#* as *+23*; and *=* as *+3D*. Also note that modern email
services will expect userids and passwords containing non-ASCII characters to Base64 encoding can be used instead of xtext encoding for the user identifier
use UTF-8 encoding with RFC-4013_ normalisation applied. and plain password by replacing *plain* by *plain:b*.
Note that modern email services will expect userids and passwords containing
non-ASCII characters to use UTF-8 encoding with RFC-4013_ normalisation applied.
Authentication proceeds according to an authentication 'mechanism' that is Authentication proceeds according to an authentication 'mechanism' that is
advertised by the server and selected by the client. Many authentication advertised by the server and selected by the client. Many authentication
@ -693,10 +698,10 @@ available via PAM (see below).
The PLAIN, LOGIN and CRAM-MD5 mechanisms can use plaintext passwords, stored The PLAIN, LOGIN and CRAM-MD5 mechanisms can use plaintext passwords, stored
in the secrets file using a password-type of *plain*. In addition, the in the secrets file using a password-type of *plain*. In addition, the
CRAM-MD5 mechanism can also use hashed passwords generated by the CRAM-MD5 mechanism can also make use of hashed passwords generated by the
*emailrelay-passwd* program and these are stored in the secrets file with a *emailrelay-passwd* program and these are stored in the secrets file with a
password-type of *md5*. (Hashed passwords are marginally more secure because password-type of *md5*. (Hashed passwords are marginally more secure because
the plaintext password which might be used on other accounts, is not easily the plaintext password which might be used on other accounts is not easily
recovered. However, hashed passwords can only be used for HMAC authentication recovered. However, hashed passwords can only be used for HMAC authentication
mechanisms that are based on the same hash function.) The XOAUTH2 mechanism mechanisms that are based on the same hash function.) The XOAUTH2 mechanism
can be used for client-side authentication using tokens that have been can be used for client-side authentication using tokens that have been
@ -774,10 +779,10 @@ described below.
TLS encryption TLS encryption
============== ==============
E-MailRelay can use negotiated TLS to encrypt SMTP and POP sessions: to enable E-MailRelay can use negotiated TLS to encrypt SMTP and POP sessions: use the
client-side TLS encryption when E-MailRelay is acting as an SMTP client use the *--client-tls* command-line option to enable client-side TLS encryption when
*--client-tls* command-line option, and to enable server-side TLS when E-MailRelay is acting as an SMTP client, and use *--server-tls* to enable
E-MailRelay is acting as an SMTP or POP server use *--server-tls*. The server-side TLS when E-MailRelay is acting as an SMTP or POP server. The
connections start off as unencrypted and the SMTP command *STARTTLS* (or the connections start off as unencrypted and the SMTP command *STARTTLS* (or the
POP *STLS* command) can be used to negotiate TLS encryption before any POP *STLS* command) can be used to negotiate TLS encryption before any
passwords are exchanged. passwords are exchanged.
@ -1289,9 +1294,14 @@ simple command-line interface which is compatible with *netcat* and *telnet*:
E-MailRelay> help E-MailRelay> help
E-MailRelay> quit E-MailRelay> quit
The *flush* command is used to get the E-MailRelay server to forward spooled The *forward* command is used to trigger the E-MailRelay server into forwarding
mail to the next SMTP server. The *forward* command does the same but without spooled mail to the next SMTP server.
waiting for completion.
The *flush* command is similar but it uses its own connection to the SMTP
server and waits for the messages to be sent.
The *unfail-all* command can be used to remove the *.bad* filename extension
from files in the spool directory.
The *list* command lists the messages in the spool directory, *status* provides The *list* command lists the messages in the spool directory, *status* provides
network status information and activity statistics, and *notify* enables network status information and activity statistics, and *notify* enables
@ -1351,7 +1361,7 @@ Standard (FHS_) use this configure command:
It is possible to change the installation root directory after building by It is possible to change the installation root directory after building by
using *make DESTDIR=<root> install* or *DESTDIR=<root> make -e install*. using *make DESTDIR=<root> install* or *DESTDIR=<root> make -e install*.
However, this will not change the default spool directory path built into the However, this will not change the default spool directory path built into the
scripts and executables so the correct spool directory will have to be scripts and executables so the correct spool directory will then have to be
specified at run-time with the *--spool-dir* command-line option. specified at run-time with the *--spool-dir* command-line option.
On Windows the installation GUI prompts for two installation directories, On Windows the installation GUI prompts for two installation directories,

View File

@ -178,7 +178,7 @@ where <option> is:
# --hidden (-H) # --hidden (-H)
Windows only. Hides the application window and disables all message boxes, Windows only. Hides the application window and disables all message boxes,
overriding any "--show" option. This is useful when running as a windows overriding any "--show" option. This is useful when running as a windows
service. service.
# --idle-timeout <time> # --idle-timeout <time>
@ -205,8 +205,9 @@ where <option> is:
suffix only their IPv4 or IPv6 addresses will be used (eg. "ppp0-ipv4"). suffix only their IPv4 or IPv6 addresses will be used (eg. "ppp0-ipv4").
# --localedir <dir> # --localedir <dir>
Specifies a locale base directory where localisation message catalogues can Enables localisation and specifies the locale base directory where message
be found. An empty directory can be used for the built-in default. catalogues can be found. An empty directory can be used for the built-in
default.
# --log (-l) # --log (-l)
Enables logging to the standard error stream and to the syslog. The Enables logging to the standard error stream and to the syslog. The
@ -357,9 +358,10 @@ where <option> is:
to set a maximum version. to set a maximum version.
# --user <username> (-u) # --user <username> (-u)
When started as root the program switches to an non-privileged effective When started as root the program switches to a non-privileged effective
user-id when idle. This option can be used to define which user-id is used. user-id when idle. This option can be used to define the idle user-id and
Specify "root" to disable all user-id switching. Ignored on Windows. also the group ownership of new files and sockets. Specify "root" to
disable all user-id switching. Ignored on Windows.
# --verbose (-v) # --verbose (-v)
Enables more verbose logging when used with "--log", and more verbose help Enables more verbose logging when used with "--log", and more verbose help
@ -407,7 +409,7 @@ command-line options:
* as each message is submitted, just before receipt is acknowledged ("--immediate") * as each message is submitted, just before receipt is acknowledged ("--immediate")
* as soon as the submitting client connection disconnects ("--forward-on-disconnect") * as soon as the submitting client connection disconnects ("--forward-on-disconnect")
* periodically ("--poll=<seconds>") * periodically ("--poll=<seconds>")
* on demand using the administration interface's "flush" command ("--admin=<port>") * on demand using the administration interface's "forward" command ("--admin=<port>")
* when a "--filter" script exits with an exit code of 103 * when a "--filter" script exits with an exit code of 103
These can be mixed. These can be mixed.
@ -592,13 +594,16 @@ user identifier; and the "password" field is the xtext-encoded plain password
or a base64-encoded "HMAC-MD5" state. For "client" lines the password-type can or a base64-encoded "HMAC-MD5" state. For "client" lines the password-type can
also be "oauth". also be "oauth".
The first two fields are case-insensitive. The "xtext" encoding scheme is The "xtext" encoding scheme is defined properly in RFC-3461, but basically it
defined properly in RFC-3461, but basically it says that non-alphanumeric says that non-alphanumeric characters (including space, "+", "#" and "=") should
characters (including space, "+", "#" and "=") should be represented in be represented in uppercase hexadecimal ascii as "+XX". So a space should be
uppercase hexadecimal ascii as "+XX". So a space should be written as "+20"; written as "+20"; "+" as "+2B"; "#" as "+23"; and "=" as "+3D".
"+" as "+2B"; "#" as "+23"; and "=" as "+3D". Also note that modern email
services will expect userids and passwords containing non-ASCII characters to Base64 encoding can be used instead of xtext encoding for the user identifier
use UTF-8 encoding with RFC-4013 normalisation applied. and plain password by replacing "plain" by "plain:b".
Note that modern email services will expect userids and passwords containing
non-ASCII characters to use UTF-8 encoding with RFC-4013 normalisation applied.
Authentication proceeds according to an authentication 'mechanism' that is Authentication proceeds according to an authentication 'mechanism' that is
advertised by the server and selected by the client. Many authentication advertised by the server and selected by the client. Many authentication
@ -610,10 +615,10 @@ available via PAM (see below).
The PLAIN, LOGIN and CRAM-MD5 mechanisms can use plaintext passwords, stored The PLAIN, LOGIN and CRAM-MD5 mechanisms can use plaintext passwords, stored
in the secrets file using a password-type of "plain". In addition, the in the secrets file using a password-type of "plain". In addition, the
CRAM-MD5 mechanism can also use hashed passwords generated by the CRAM-MD5 mechanism can also make use of hashed passwords generated by the
"emailrelay-passwd" program and these are stored in the secrets file with a "emailrelay-passwd" program and these are stored in the secrets file with a
password-type of "md5". (Hashed passwords are marginally more secure because password-type of "md5". (Hashed passwords are marginally more secure because
the plaintext password which might be used on other accounts, is not easily the plaintext password which might be used on other accounts is not easily
recovered. However, hashed passwords can only be used for HMAC authentication recovered. However, hashed passwords can only be used for HMAC authentication
mechanisms that are based on the same hash function.) The XOAUTH2 mechanism mechanisms that are based on the same hash function.) The XOAUTH2 mechanism
can be used for client-side authentication using tokens that have been can be used for client-side authentication using tokens that have been
@ -685,10 +690,10 @@ described below.
TLS encryption TLS encryption
-------------- --------------
E-MailRelay can use negotiated TLS to encrypt SMTP and POP sessions: to enable E-MailRelay can use negotiated TLS to encrypt SMTP and POP sessions: use the
client-side TLS encryption when E-MailRelay is acting as an SMTP client use the "--client-tls" command-line option to enable client-side TLS encryption when
"--client-tls" command-line option, and to enable server-side TLS when E-MailRelay is acting as an SMTP client, and use "--server-tls" to enable
E-MailRelay is acting as an SMTP or POP server use "--server-tls". The server-side TLS when E-MailRelay is acting as an SMTP or POP server. The
connections start off as unencrypted and the SMTP command "STARTTLS" (or the connections start off as unencrypted and the SMTP command "STARTTLS" (or the
POP "STLS" command) can be used to negotiate TLS encryption before any POP "STLS" command) can be used to negotiate TLS encryption before any
passwords are exchanged. passwords are exchanged.
@ -1135,9 +1140,14 @@ simple command-line interface which is compatible with "netcat" and "telnet":
E-MailRelay> help E-MailRelay> help
E-MailRelay> quit E-MailRelay> quit
The "flush" command is used to get the E-MailRelay server to forward spooled The "forward" command is used to trigger the E-MailRelay server into forwarding
mail to the next SMTP server. The "forward" command does the same but without spooled mail to the next SMTP server.
waiting for completion.
The "flush" command is similar but it uses its own connection to the SMTP
server and waits for the messages to be sent.
The "unfail-all" command can be used to remove the ".bad" filename extension
from files in the spool directory.
The "list" command lists the messages in the spool directory, "status" provides The "list" command lists the messages in the spool directory, "status" provides
network status information and activity statistics, and "notify" enables network status information and activity statistics, and "notify" enables
@ -1194,7 +1204,7 @@ Standard (FHS) use this configure command:
It is possible to change the installation root directory after building by It is possible to change the installation root directory after building by
using "make DESTDIR=<root> install" or "DESTDIR=<root> make -e install". using "make DESTDIR=<root> install" or "DESTDIR=<root> make -e install".
However, this will not change the default spool directory path built into the However, this will not change the default spool directory path built into the
scripts and executables so the correct spool directory will have to be scripts and executables so the correct spool directory will then have to be
specified at run-time with the "--spool-dir" command-line option. specified at run-time with the "--spool-dir" command-line option.
On Windows the installation GUI prompts for two installation directories, On Windows the installation GUI prompts for two installation directories,

View File

@ -20,6 +20,11 @@
into protected directories like <em>Program Files</em>. into protected directories like <em>Program Files</em>.
</p> </p>
<p>
You may need to run <em>vc_redist.x64.exe</em> first to install the Microsoft C++
run-time files.
</p>
<p> <p>
The setup GUI will take you through the installation options and then install The setup GUI will take you through the installation options and then install
the run-time files into your chosen locations. the run-time files into your chosen locations.

View File

@ -9,6 +9,9 @@ setup program `emailrelay-setup.exe` and its associated `payload` files.
Run `emailrelay-setup.exe` as an administrator if you are going to be installing Run `emailrelay-setup.exe` as an administrator if you are going to be installing
into protected directories like `Program Files`. into protected directories like `Program Files`.
You may need to run `vc_redist.x64.exe` first to install the Microsoft C++
run-time files.
The setup GUI will take you through the installation options and then install The setup GUI will take you through the installation options and then install
the run-time files into your chosen locations. the run-time files into your chosen locations.

View File

@ -10,6 +10,9 @@ setup program *emailrelay-setup.exe* and its associated *payload* files.
Run *emailrelay-setup.exe* as an administrator if you are going to be installing Run *emailrelay-setup.exe* as an administrator if you are going to be installing
into protected directories like *Program Files*. into protected directories like *Program Files*.
You may need to run *vc_redist.x64.exe* first to install the Microsoft C++
run-time files.
The setup GUI will take you through the installation options and then install The setup GUI will take you through the installation options and then install
the run-time files into your chosen locations. the run-time files into your chosen locations.

View File

@ -1,11 +1,11 @@
Summary: Simple e-mail message transfer agent and proxy using SMTP Summary: Simple e-mail message transfer agent and proxy using SMTP
Name: emailrelay Name: emailrelay
Version: 2.2.1 Version: 2.3
Release: 1 Release: 1
License: GPL3 License: GPL3
Group: System Environment/Daemons Group: System Environment/Daemons
URL: http://emailrelay.sourceforge.net URL: http://emailrelay.sourceforge.net
Source: http://sourceforge.net/projects/emailrelay/files/emailrelay/2.2/emailrelay-2.2.1-src.tar.gz Source: http://sourceforge.net/projects/emailrelay/files/emailrelay/2.3/emailrelay-2.3-src.tar.gz
BuildRoot: /tmp/emailrelay-install BuildRoot: /tmp/emailrelay-install
%description %description

View File

@ -63,7 +63,7 @@ install-data-hook:
if test -f "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" ; then : ; else sed -e "s:^#spool-dir .*:spool-dir $(e_spooldir):g" -e 's:"/var/spool/emailrelay":"'"$(e_spooldir)"'":g' -e "s:/etc:$(e_sysconfdir):g" -e "s:/usr/local/bin:$(e_libexecdir):g" < "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf.template" > "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" ; fi if test -f "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" ; then : ; else sed -e "s:^#spool-dir .*:spool-dir $(e_spooldir):g" -e 's:"/var/spool/emailrelay":"'"$(e_spooldir)"'":g' -e "s:/etc:$(e_sysconfdir):g" -e "s:/usr/local/bin:$(e_libexecdir):g" < "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf.template" > "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" ; fi
cp "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf.makeinstall" cp "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf.makeinstall"
# remove the .conf file if it has not been editied # remove the .conf file if it has not been edited
uninstall-hook: uninstall-hook:
if diff -q "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf.makeinstall" ; then rm "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" ; fi if diff -q "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf.makeinstall" ; then rm "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" ; fi
-rm "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf.makeinstall" -rm "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf.makeinstall"

View File

@ -606,7 +606,7 @@ emailrelay.service: emailrelay.service.in
@GCONFIG_INSTALL_HOOK_TRUE@ if test -f "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" ; then : ; else sed -e "s:^#spool-dir .*:spool-dir $(e_spooldir):g" -e 's:"/var/spool/emailrelay":"'"$(e_spooldir)"'":g' -e "s:/etc:$(e_sysconfdir):g" -e "s:/usr/local/bin:$(e_libexecdir):g" < "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf.template" > "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" ; fi @GCONFIG_INSTALL_HOOK_TRUE@ if test -f "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" ; then : ; else sed -e "s:^#spool-dir .*:spool-dir $(e_spooldir):g" -e 's:"/var/spool/emailrelay":"'"$(e_spooldir)"'":g' -e "s:/etc:$(e_sysconfdir):g" -e "s:/usr/local/bin:$(e_libexecdir):g" < "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf.template" > "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" ; fi
@GCONFIG_INSTALL_HOOK_TRUE@ cp "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf.makeinstall" @GCONFIG_INSTALL_HOOK_TRUE@ cp "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf.makeinstall"
# remove the .conf file if it has not been editied # remove the .conf file if it has not been edited
@GCONFIG_INSTALL_HOOK_TRUE@uninstall-hook: @GCONFIG_INSTALL_HOOK_TRUE@uninstall-hook:
@GCONFIG_INSTALL_HOOK_TRUE@ if diff -q "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf.makeinstall" ; then rm "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" ; fi @GCONFIG_INSTALL_HOOK_TRUE@ if diff -q "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf.makeinstall" ; then rm "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf" ; fi
@GCONFIG_INSTALL_HOOK_TRUE@ -rm "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf.makeinstall" @GCONFIG_INSTALL_HOOK_TRUE@ -rm "$(DESTDIR)$(e_sysconfdir)/emailrelay.conf.makeinstall"

View File

@ -36,5 +36,8 @@
# "password" fields should be encoded using the RFC-1891 "xtext" encoding # "password" fields should be encoded using the RFC-1891 "xtext" encoding
# scheme so that they are representing as hexadecimal ascii as "+XX". # scheme so that they are representing as hexadecimal ascii as "+XX".
# #
# Alternatively use "plain:b" instead of "plain" and then use Base64 encoding
# for the id and password.
#
# The "none" rows allow trusted IP addresses to bypass authentication. # The "none" rows allow trusted IP addresses to bypass authentication.
# #

View File

@ -277,8 +277,8 @@
# Name: hidden # Name: hidden
# Format: hidden # Format: hidden
# Description: Windows only. Hides the application window and disables all # Description: Windows only. Hides the application window and disables all
# message boxes, overriding any ""--show"" option. This is useful when # message boxes, overriding any "--show" option. This is useful when
# running as a windows service. # running as a windows service.
# #
#hidden #hidden
@ -317,9 +317,9 @@
# Name: localedir # Name: localedir
# Format: localedir <dir> # Format: localedir <dir>
# Description: Specifies a locale base directory where localisation message # Description: Enables localisation and specifies the locale base directory
# catalogues can be found. An empty directory can be used for the built-in # where message catalogues can be found. An empty directory can be used for
# default. # the built-in default.
# #
#localedir /opt/share/locale #localedir /opt/share/locale
@ -568,10 +568,10 @@
# Name: user # Name: user
# Format: user <username> # Format: user <username>
# Description: When started as root the program switches to an non-privileged # Description: When started as root the program switches to a non-privileged
# effective user-id when idle. This option can be used to define which # effective user-id when idle. This option can be used to define the idle
# user-id is used. Specify "root" to disable all user-id switching. Ignored # user-id and also the group ownership of new files and sockets. Specify
# on Windows. # "root" to disable all user-id switching. Ignored on Windows.
# #
#user nobody #user nobody

View File

@ -138,11 +138,10 @@
./src/gnet/glocation.cpp ./src/gnet/glocation.cpp
./src/gnet/gtimer.cpp ./src/gnet/gtimer.cpp
./src/gnet/gtimerlist.cpp ./src/gnet/gtimerlist.cpp
./src/gnet/gaddress_ipv6.cpp ./src/gnet/gaddress.cpp
./src/gnet/gsocket_unix.cpp ./src/gnet/gsocket_unix.cpp
./src/gnet/geventloggingcontext.cpp ./src/gnet/geventloggingcontext.cpp
./src/gnet/gconnection.cpp ./src/gnet/gconnection.cpp
./src/gnet/gaddress_ipv4.cpp
./src/gnet/gclient.cpp ./src/gnet/gclient.cpp
./src/gnet/gexceptionhandler.cpp ./src/gnet/gexceptionhandler.cpp
./src/gnet/gexceptionsource.cpp ./src/gnet/gexceptionsource.cpp

View File

@ -5,6 +5,8 @@
# #
Checks: "\ Checks: "\
bugprone-*,\ bugprone-*,\
-bugprone-easily-swappable-parameters,\
-bugprone-throw-keyword-missing,\
-bugprone-macro-parentheses,\ -bugprone-macro-parentheses,\
-bugprone-branch-clone,\ -bugprone-branch-clone,\
cert-*,\ cert-*,\
@ -20,7 +22,9 @@ cppcoreguidelines-pro-type-member-init,\
-cppcoreguidelines-avoid-magic-numbers,\ -cppcoreguidelines-avoid-magic-numbers,\
-cppcoreguidelines-macro-usage,\ -cppcoreguidelines-macro-usage,\
-cppcoreguidelines-avoid-non-const-global-variables,\ -cppcoreguidelines-avoid-non-const-global-variables,\
-cppcoreguidelines-prefer-member-initializer,\
modernize-*,\ modernize-*,\
-modernize-avoid-bind,\
-modernize-use-auto,\ -modernize-use-auto,\
-modernize-use-default-member-init,\ -modernize-use-default-member-init,\
-modernize-return-braced-init-list,\ -modernize-return-braced-init-list,\
@ -28,7 +32,9 @@ modernize-*,\
-modernize-raw-string-literal,\ -modernize-raw-string-literal,\
-modernize-use-trailing-return-type,\ -modernize-use-trailing-return-type,\
performance-*,\ performance-*,\
-performance-unnecessary-value-param,\
readability-*,\ readability-*,\
-readability-function-cognitive-complexity,\
-readability-qualified-auto,\ -readability-qualified-auto,\
-readability-simplify-boolean-expr,\ -readability-simplify-boolean-expr,\
-readability-else-after-return,\ -readability-else-after-return,\

View File

@ -39,7 +39,7 @@ namespace GAuth
GSsl::Library & lib() GSsl::Library & lib()
{ {
GSsl::Library * p = GSsl::Library::instance() ; GSsl::Library * p = GSsl::Library::instance() ;
if( p == nullptr ) throw std::runtime_error( "no tsl library" ) ; if( p == nullptr ) throw G::Exception( "no tls library" ) ;
return *p ; return *p ;
} }
struct DigesterAdaptor /// Used by GAuth::Cram to use GSsl::Digester. struct DigesterAdaptor /// Used by GAuth::Cram to use GSsl::Digester.

View File

@ -112,15 +112,15 @@ std::string GAuth::SaslClientImp::mechanism( const G::StringArray & server_mecha
our_list.push_back( LOGIN ) ; our_list.push_back( LOGIN ) ;
} }
// use the configuration string as a mechanism whitelist and/or blacklist // use the configuration string as a mechanism whitelist and/or blocklist
if( !m_config.empty() ) if( !m_config.empty() )
{ {
bool simple = G::Str::imatch( our_list , m_config ) ; // eg. allow "plain" as well as "m:plain" bool simple = G::Str::imatch( our_list , m_config ) ; // eg. allow "plain" as well as "m:plain"
G::StringArray list = G::Str::splitIntoTokens( G::Str::upper(m_config) , ";" ) ; G::StringArray list = G::Str::splitIntoTokens( G::Str::upper(m_config) , ";" ) ;
G::StringArray whitelist = G::Str::splitIntoTokens( simple?G::Str::upper(m_config): G::Str::headMatchResidue( list , "M:" ) , "," ) ; G::StringArray whitelist = G::Str::splitIntoTokens( simple?G::Str::upper(m_config): G::Str::headMatchResidue( list , "M:" ) , "," ) ;
G::StringArray blacklist = G::Str::splitIntoTokens( G::Str::headMatchResidue( list , "X:" ) , "," ) ; G::StringArray blocklist = G::Str::splitIntoTokens( G::Str::headMatchResidue( list , "X:" ) , "," ) ;
our_list.erase( G::Str::keepMatch( our_list.begin() , our_list.end() , whitelist , true ) , our_list.end() ) ; our_list.erase( G::Str::keepMatch( our_list.begin() , our_list.end() , whitelist , true ) , our_list.end() ) ;
our_list.erase( G::Str::removeMatch( our_list.begin() , our_list.end() , blacklist , true ) , our_list.end() ) ; our_list.erase( G::Str::removeMatch( our_list.begin() , our_list.end() , blocklist , true ) , our_list.end() ) ;
} }
// build the list of mechanisms that we can use with the server // build the list of mechanisms that we can use with the server

View File

@ -25,6 +25,7 @@
#include "gsaslclientsecrets.h" #include "gsaslclientsecrets.h"
#include "gexception.h" #include "gexception.h"
#include "gstrings.h" #include "gstrings.h"
#include <memory>
namespace GAuth namespace GAuth
{ {
@ -73,8 +74,8 @@ public:
///< Returns the empty string if none is supported or if not active(). ///< Returns the empty string if none is supported or if not active().
bool next() ; bool next() ;
///< Moves to the next preferred mechanism. Returns the empty ///< Moves to the next preferred mechanism. Returns false if there
///< string if there are no more mechanisms. ///< are no more mechanisms.
std::string next( const std::string & ) ; std::string next( const std::string & ) ;
///< A convenience overload that moves to the next() mechanism ///< A convenience overload that moves to the next() mechanism

View File

@ -33,13 +33,14 @@ namespace GAuth
//| \class GAuth::SaslClientSecrets //| \class GAuth::SaslClientSecrets
/// An interface used by GAuth::SaslClient to obtain a client id and /// An interface used by GAuth::SaslClient to obtain a client id and
/// its authentication secret. Conceptually there is one client and /// its authentication secret. Conceptually there is one client and
/// they can have a secrets encoded in multiple ways. /// they can have secrets encoded in multiple ways.
/// ///
class GAuth::SaslClientSecrets : public virtual Valid class GAuth::SaslClientSecrets : public virtual Valid
{ {
public: public:
virtual Secret clientSecret( const std::string & encoding_type ) const = 0 ; virtual Secret clientSecret( const std::string & type ) const = 0 ;
///< Returns the client secret for the given encoding type. ///< Returns the client secret for the given type.
///< The type is "plain" or the CRAM hash algorithm or "oauth".
///< Returns an invalid secret if none. ///< Returns an invalid secret if none.
} ; } ;

View File

@ -45,9 +45,11 @@ public:
std::string id() const ; std::string id() const ;
bool authenticated() const ; bool authenticated() const ;
private: public:
SaslServerPamImp( const SaslServerPamImp & ) ; SaslServerPamImp( const SaslServerPamImp & ) = delete ;
void operator=( const SaslServerPamImp & ) ; SaslServerPamImp( SaslServerPamImp && ) = delete ;
void operator=( const SaslServerPamImp & ) = delete ;
void operator=( SaslServerPamImp && ) = delete ;
private: private:
bool m_active ; bool m_active ;
@ -76,9 +78,11 @@ protected:
void converse( ItemArray & ) override ; void converse( ItemArray & ) override ;
void delay( unsigned int usec ) override ; void delay( unsigned int usec ) override ;
private: public:
PamImp( const PamImp & ) ; PamImp( const PamImp & ) = delete ;
void operator=( const PamImp & ) ; PamImp( PamImp && ) = delete ;
void operator=( const PamImp & ) = delete ;
void operator=( PamImp && ) = delete ;
private: private:
std::string m_app ; std::string m_app ;
@ -159,7 +163,7 @@ bool GAuth::SaslServerPamImp::init( const std::string & mechanism )
std::string GAuth::SaslServerPamImp::id() const std::string GAuth::SaslServerPamImp::id() const
{ {
return m_pam.get() ? m_pam->id() : std::string() ; return m_pam ? m_pam->id() : std::string() ;
} }
std::string GAuth::SaslServerPamImp::apply( const std::string & response , bool & done ) std::string GAuth::SaslServerPamImp::apply( const std::string & response , bool & done )

View File

@ -53,11 +53,11 @@ public:
///< Constructor. ///< Constructor.
public: public:
~SaslServerPam() ; ~SaslServerPam() override ;
SaslServerPam( const SaslServerPam & ) = delete ; SaslServerPam( const SaslServerPam & ) = delete ;
SaslServerPam( SaslServerPam && ) = delete ; SaslServerPam( SaslServerPam && ) = delete ;
void operator=( const SaslServerPam & ) = delete ; SaslServerPam & operator=( const SaslServerPam & ) = delete ;
void operator=( SaslServerPam && ) = delete ; SaslServerPam & operator=( SaslServerPam && ) = delete ;
private: // overrides private: // overrides
bool requiresEncryption() const override ; // Override from GAuth::SaslServer. bool requiresEncryption() const override ; // Override from GAuth::SaslServer.

View File

@ -39,8 +39,9 @@ namespace GAuth
class GAuth::SaslServerSecrets : public virtual Valid class GAuth::SaslServerSecrets : public virtual Valid
{ {
public: public:
virtual Secret serverSecret( const std::string & encoding_type , const std::string & id ) const = 0 ; virtual Secret serverSecret( const std::string & type , const std::string & id ) const = 0 ;
///< Returns the server secret for the given client id. ///< Returns the server secret for the given client id.
///< The type is "plain" or the CRAM hash algorithm.
///< Returns an invalid secret if not found. ///< Returns an invalid secret if not found.
virtual std::pair<std::string,std::string> serverTrust( const std::string & address_range ) const = 0 ; virtual std::pair<std::string,std::string> serverTrust( const std::string & address_range ) const = 0 ;

View File

@ -150,7 +150,8 @@ bool GAuth::Secret::isDotted( const std::string & s )
std::string GAuth::Secret::undotted( const std::string & s ) std::string GAuth::Secret::undotted( const std::string & s )
{ {
G::StringArray decimals = G::Str::splitIntoFields( s , "." ) ; decimals.resize( 8U ) ; G::StringArray decimals = G::Str::splitIntoFields( s , "." ) ;
decimals.resize( 8U ) ;
std::string result ; std::string result ;
for( std::size_t i = 0U ; i < 8U ; i++ ) for( std::size_t i = 0U ; i < 8U ; i++ )
{ {

View File

@ -37,18 +37,18 @@ void GAuth::Secrets::check( const std::string & p1 , const std::string & p2 , co
std::for_each( list.begin() , list.end() , &SecretsFile::check ) ; std::for_each( list.begin() , list.end() , &SecretsFile::check ) ;
} }
GAuth::Secrets::Secrets( const std::string & path , const std::string & name , const std::string & type ) : GAuth::Secrets::Secrets( const std::string & path , const std::string & name ) :
m_source(path) m_source(path)
{ {
G_DEBUG( "GAuth::Secrets:ctor: [" << path << "]" ) ; G_DEBUG( "GAuth::Secrets:ctor: [" << path << "]" ) ;
if( m_source != "/pam" ) if( m_source != "/pam" )
m_imp = std::make_unique<SecretsFile>( path , true , name , type ) ; m_imp = std::make_unique<SecretsFile>( path , true , name ) ;
} }
GAuth::Secrets::Secrets() GAuth::Secrets::Secrets()
{ {
if( m_source != "/pam" ) if( m_source != "/pam" )
m_imp = std::make_unique<SecretsFile>( std::string() , true , std::string() , std::string() ) ; m_imp = std::make_unique<SecretsFile>( std::string() , false , std::string() ) ;
} }
GAuth::Secrets::~Secrets() GAuth::Secrets::~Secrets()
@ -64,14 +64,14 @@ bool GAuth::Secrets::valid() const
return m_source == "/pam" || m_imp->valid() ; return m_source == "/pam" || m_imp->valid() ;
} }
GAuth::Secret GAuth::Secrets::clientSecret( const std::string & mechanism ) const GAuth::Secret GAuth::Secrets::clientSecret( const std::string & type ) const
{ {
return valid() ? m_imp->clientSecret(mechanism) : Secret::none() ; return valid() ? m_imp->clientSecret(type) : Secret::none() ;
} }
GAuth::Secret GAuth::Secrets::serverSecret( const std::string & mechanism , const std::string & id ) const GAuth::Secret GAuth::Secrets::serverSecret( const std::string & type , const std::string & id ) const
{ {
return valid() ? m_imp->serverSecret( mechanism , id ) : Secret::none() ; return valid() ? m_imp->serverSecret( type , id ) : Secret::none() ;
} }
std::pair<std::string,std::string> GAuth::Secrets::serverTrust( const std::string & address_range ) const std::pair<std::string,std::string> GAuth::Secrets::serverTrust( const std::string & address_range ) const
@ -79,8 +79,8 @@ std::pair<std::string,std::string> GAuth::Secrets::serverTrust( const std::strin
return valid() ? m_imp->serverTrust( address_range ) : std::make_pair(std::string(),std::string()) ; return valid() ? m_imp->serverTrust( address_range ) : std::make_pair(std::string(),std::string()) ;
} }
bool GAuth::Secrets::contains( const std::string & mechanism ) const bool GAuth::Secrets::contains( const std::string & type ) const
{ {
return valid() ? m_imp->contains( mechanism ) : false ; return valid() ? m_imp->contains( type ) : false ;
} }

View File

@ -46,21 +46,15 @@ public:
///< Checks the given secret sources. Logs warnings and throws ///< Checks the given secret sources. Logs warnings and throws
///< an exception if there are any fatal errors. ///< an exception if there are any fatal errors.
Secrets( const std::string & source_storage_path , const std::string & debug_name , Secrets( const std::string & source_storage_path , const std::string & debug_name ) ;
const std::string & server_type = std::string() ) ; ///< Constructor. The connection string is a secrets file path
///< Constructor. The connection string is a secrets file path ///< or "/pam".
///< or "/pam". ///<
///< ///< The 'debug-name' is used in log and error messages to
///< The 'debug-name' is used in log and error messages to ///< identify the repository.
///< identify the repository. ///<
///< ///< Throws on error, although an empty path is not
///< The 'server-type' parameter can be used to select ///< considered an error: see valid().
///< a different set of server-side authentication records
///< that may be stored in the same repository such as
///< "smtp" or "pop". The default is "server".
///<
///< Throws on error, although an empty path is not
///< considered an error: see valid().
Secrets() ; Secrets() ;
///< Default constructor for an in-valid(), empty-path object. ///< Default constructor for an in-valid(), empty-path object.
@ -68,10 +62,10 @@ public:
bool valid() const override ; bool valid() const override ;
///< Override from GAuth::Valid virtual base. ///< Override from GAuth::Valid virtual base.
Secret serverSecret( const std::string & mechanism , const std::string & id ) const override ; Secret serverSecret( const std::string & type , const std::string & id ) const override ;
///< Override from GAuth::SaslServerSecrets. ///< Override from GAuth::SaslServerSecrets.
bool contains( const std::string & mechanism ) const override ; bool contains( const std::string & type ) const override ;
///< Override from GAuth::SaslServerSecrets. ///< Override from GAuth::SaslServerSecrets.
public: public:
@ -83,7 +77,7 @@ public:
private: // overrides private: // overrides
std::string source() const override ; // Override from GAuth::SaslServerSecrets. std::string source() const override ; // Override from GAuth::SaslServerSecrets.
Secret clientSecret( const std::string & mechanism ) const override ; // Override from GAuth::SaslClientSecrets. Secret clientSecret( const std::string & type ) const override ; // Override from GAuth::SaslClientSecrets.
std::pair<std::string,std::string> serverTrust( const std::string & address_range ) const override ; // Override from GAuth::SaslServerSecrets. std::pair<std::string,std::string> serverTrust( const std::string & address_range ) const override ; // Override from GAuth::SaslServerSecrets.
private: private:

View File

@ -23,6 +23,7 @@
#include "gsecrets.h" #include "gsecrets.h"
#include "groot.h" #include "groot.h"
#include "gxtext.h" #include "gxtext.h"
#include "gbase64.h"
#include "gstr.h" #include "gstr.h"
#include "gdatetime.h" #include "gdatetime.h"
#include "gfile.h" #include "gfile.h"
@ -35,31 +36,23 @@
#include <utility> // std::swap(), std::pair #include <utility> // std::swap(), std::pair
#include <sstream> #include <sstream>
GAuth::SecretsFile::SecretsFile( const G::Path & path , bool auto_reread , const std::string & debug_name , GAuth::SecretsFile::SecretsFile( const G::Path & path , bool auto_reread , const std::string & debug_name ) :
const std::string & server_type ) : m_path(path) ,
m_path(path) , m_auto(auto_reread) ,
m_auto(auto_reread) , m_debug_name(debug_name) ,
m_debug_name(debug_name) , m_file_time(0) ,
m_server_type(G::Str::lower(server_type)) , m_check_time(G::SystemTime::now())
m_file_time(0) ,
m_check_time(G::SystemTime::now())
{ {
m_server_type = m_server_type.empty() ? std::string("server") : m_server_type ;
m_valid = ! path.str().empty() ; m_valid = ! path.str().empty() ;
if( m_valid ) if( m_valid )
read( path , false ) ; read( path ) ;
} }
void GAuth::SecretsFile::check( const std::string & path ) void GAuth::SecretsFile::check( const std::string & path )
{
checkImp( path , true , "server" ) ; // allow only 'client' or 'server' lines
}
void GAuth::SecretsFile::checkImp( const std::string & path , bool strict_side , const std::string & server_type )
{ {
if( !path.empty() ) if( !path.empty() )
{ {
Contents contents = readContents( path , server_type , strict_side ) ; Contents contents = readContents( path ) ;
showWarnings( contents.m_warnings , path ) ; showWarnings( contents.m_warnings , path ) ;
if( !contents.m_warnings.empty() ) if( !contents.m_warnings.empty() )
throw Error() ; throw Error() ;
@ -90,16 +83,16 @@ void GAuth::SecretsFile::reread( int )
if( t != m_file_time ) if( t != m_file_time )
{ {
G_LOG_S( "GAuth::Secrets: re-reading secrets file: " << m_path ) ; G_LOG_S( "GAuth::Secrets: re-reading secrets file: " << m_path ) ;
read( m_path , false ) ; read( m_path ) ;
} }
} }
} }
} }
void GAuth::SecretsFile::read( const G::Path & path , bool strict_side ) void GAuth::SecretsFile::read( const G::Path & path )
{ {
m_file_time = readFileTime( path ) ; m_file_time = readFileTime( path ) ;
m_contents = readContents( path , m_server_type , strict_side ) ; m_contents = readContents( path ) ;
showWarnings( m_contents.m_warnings , path , m_debug_name ) ; showWarnings( m_contents.m_warnings , path , m_debug_name ) ;
} }
@ -109,7 +102,7 @@ G::SystemTime GAuth::SecretsFile::readFileTime( const G::Path & path )
return G::File::time( path ) ; return G::File::time( path ) ;
} }
GAuth::SecretsFile::Contents GAuth::SecretsFile::readContents( const G::Path & path , const std::string & server_type , bool strict_side ) GAuth::SecretsFile::Contents GAuth::SecretsFile::readContents( const G::Path & path )
{ {
std::unique_ptr<std::ifstream> file ; std::unique_ptr<std::ifstream> file ;
{ {
@ -121,10 +114,10 @@ GAuth::SecretsFile::Contents GAuth::SecretsFile::readContents( const G::Path & p
throw Secrets::OpenError( path.str() ) ; throw Secrets::OpenError( path.str() ) ;
} }
return readContents( *file , server_type , strict_side ) ; return readContents( *file ) ;
} }
GAuth::SecretsFile::Contents GAuth::SecretsFile::readContents( std::istream & file , const std::string & server_type , bool strict_side ) GAuth::SecretsFile::Contents GAuth::SecretsFile::readContents( std::istream & file )
{ {
Contents contents ; Contents contents ;
for( unsigned int line_number = 1U ; file.good() ; ++line_number ) for( unsigned int line_number = 1U ; file.good() ; ++line_number )
@ -141,61 +134,80 @@ GAuth::SecretsFile::Contents GAuth::SecretsFile::readContents( std::istream & fi
if( word_array.size() >= 4U ) if( word_array.size() >= 4U )
{ {
// 0=<client-server> 1=<encoding-type> 2=<userid-or-ipaddress> 3=<secret-or-verifier-hint> // 0=<client-server> 1=<encoding-type> 2=<userid-or-ipaddress> 3=<secret-or-verifier-hint>
processLine( contents , server_type , line_number , word_array[0U] , word_array[1U] , word_array[2U] , word_array[3U] , strict_side ) ; processLine( contents , line_number , word_array[0U] , word_array[1U] , word_array[2U] , word_array[3U] ) ;
} }
else else
{ {
addWarning( contents , line_number , "too few fields" , "" ) ; addWarning( contents , line_number , "too few fields" ) ;
} }
} }
} }
return contents ; return contents ;
} }
void GAuth::SecretsFile::processLine( Contents & contents , const std::string & server_type , void GAuth::SecretsFile::processLine( Contents & contents ,
unsigned int line_number , const std::string & side_in , const std::string & encoding_type_in , unsigned int line_number , const std::string & side , const std::string & type_in ,
const std::string & id_xtext_or_ip , const std::string & secret , bool strict_side ) const std::string & id_or_ip_in , const std::string & secret_in )
{ {
G_ASSERT( !side_in.empty() && !encoding_type_in.empty() && !id_xtext_or_ip.empty() && !secret.empty() ) ; std::string type = G::Str::lower( type_in ) ;
std::string encoding_type = G::Str::lower( encoding_type_in ) ; std::string id_or_ip = id_or_ip_in ;
std::string side = G::Str::lower( side_in ) ; std::string secret = secret_in ;
if( type == "plain:b" )
if( !G::Xtext::valid(id_xtext_or_ip) ) // (ip address ranges are valid xtext)
addWarning( contents , line_number , "invalid xtext encoding in third field" , id_xtext_or_ip ) ;
if( encoding_type == "client" || encoding_type == "server" || encoding_type == server_type ) // old-style field order, eg. "cram-md5 server ..."
{ {
const bool plain = side == "plain" || side == "login" ; // for now just re-encode to xtext -- TODO rework secrets-file encodings
addWarning( contents , line_number , "incorrect field order: use \"" + encoding_type + " " + (plain?"plain":"md5") + " <id> <" + (plain?"pwd":"hash") + ">\"" , "" ) ; bool valid_id = G::Base64::valid( id_or_ip ) ;
bool valid_secret = G::Base64::valid( secret ) ;
if( !valid_id )
addWarning( contents , line_number , "invalid base64 encoding in third field" , id_or_ip ) ;
if( !valid_secret )
addWarning( contents , line_number , "invalid base64 encoding in fourth field" ) ;
if( !valid_id || !valid_secret )
return ;
type = "plain" ;
id_or_ip = G::Xtext::encode( G::Base64::decode(id_or_ip) ) ;
secret = G::Xtext::encode( G::Base64::decode(secret) ) ;
} }
else if( side == server_type ) processLineImp( contents , line_number , G::Str::lower(side) , type , id_or_ip , secret ) ;
}
void GAuth::SecretsFile::processLineImp( Contents & contents ,
unsigned int line_number , const std::string & side , const std::string & type ,
const std::string & id_or_ip , const std::string & secret )
{
G_ASSERT( !side.empty() && !type.empty() && !id_or_ip.empty() && !secret.empty() ) ;
if( type == "plain" )
{
if( !G::Xtext::valid(id_or_ip) ) // (ip address ranges are valid xtext)
addWarning( contents , line_number , "invalid xtext encoding in third field" , id_or_ip ) ;
if( !G::Xtext::valid(secret) )
addWarning( contents , line_number , "invalid xtext encoding in fourth field" ) ; // (new)
}
if( side == "server" )
{ {
// server-side // server-side
std::string key = serverKey( encoding_type , id_xtext_or_ip ) ; std::string key = serverKey( type , id_or_ip ) ;
Value value( secret , line_number ) ; Value value( secret , line_number ) ;
bool inserted = contents.m_map.insert(std::make_pair(key,value)).second ; bool inserted = contents.m_map.insert(std::make_pair(key,value)).second ;
if( inserted ) if( inserted )
contents.m_types.insert( canonical(encoding_type) ) ; contents.m_types.insert( canonical(type) ) ;
else else
addWarning( contents , line_number , "duplicate server secret" , key ) ; addWarning( contents , line_number , "duplicate server secret" , key ) ;
} }
else if( side == "client" ) else if( side == "client" )
{ {
// client-side // client-side
const std::string & id = id_xtext_or_ip ; const std::string & id = id_or_ip ;
std::string key = clientKey( encoding_type ) ; // not including user id std::string key = clientKey( type ) ; // not including user id
Value value( id + " " + secret , line_number ) ; Value value( id + " " + secret , line_number ) ;
bool inserted = contents.m_map.insert(std::make_pair(key,value)).second ; bool inserted = contents.m_map.insert(std::make_pair(key,value)).second ;
if( !inserted ) if( !inserted )
addWarning( contents , line_number , "too many client secrets" , key ) ; addWarning( contents , line_number , "too many client secrets" , key ) ;
} }
else if( strict_side )
{
addWarning( contents , line_number , "invalid value in first field: expecting 'client' or '" + server_type + "'" , side ) ;
}
else else
{ {
G_DEBUG( "GAuth::SecretsFile::processLine: ignoring line number " << line_number << ": not 'client' or '" << server_type << "'" ) ; addWarning( contents , line_number , "invalid value in first field" , side ) ;
} }
} }
@ -218,38 +230,40 @@ void GAuth::SecretsFile::showWarnings( const Warnings & warnings , const G::Path
} }
} }
std::string GAuth::SecretsFile::canonical( const std::string & encoding_type ) std::string GAuth::SecretsFile::canonical( const std::string & type_in )
{ {
std::string type = G::Str::lower( encoding_type ) ; // (cram-md5, apop and login are for backwards compatibility -- new
// code exects plain, md5, sha1, sha512 etc)
std::string type = G::Str::lower( type_in ) ;
if( type == "cram-md5" ) type = "md5" ; if( type == "cram-md5" ) type = "md5" ;
if( type == "apop" ) type = "md5" ; if( type == "apop" ) type = "md5" ;
if( type == "login" ) type = "plain" ; if( type == "login" ) type = "plain" ;
return type ; return type ;
} }
std::string GAuth::SecretsFile::serverKey( const std::string & encoding_type , const std::string & id_xtext ) std::string GAuth::SecretsFile::serverKey( const std::string & type , const std::string & id_xtext )
{ {
// eg. key -> value... // eg. key -> value...
// "server plain bob" -> "e+3Dmc2" // "server plain bob" -> "e+3Dmc2"
// "server md5 bob" -> "xbase64x==" // "server md5 bob" -> "xbase64x=="
// "server none 192.168.0.0/24" -> "trustee" // "server none 192.168.0.0/24" -> "trustee"
// "server none ::1/128" -> "trustee" // "server none ::1/128" -> "trustee"
return "server " + canonical(encoding_type) + " " + id_xtext ; return "server " + canonical(type) + " " + id_xtext ;
} }
std::string GAuth::SecretsFile::clientKey( const std::string & encoding_type ) std::string GAuth::SecretsFile::clientKey( const std::string & type )
{ {
// eg. key -> value... // eg. key -> value...
// "client plain" -> "alice secret+21" // "client plain" -> "alice secret+21"
// "client md5" -> "alice xbase64x==" // "client md5" -> "alice xbase64x=="
return "client " + canonical(encoding_type) ; return "client " + canonical(type) ;
} }
GAuth::Secret GAuth::SecretsFile::clientSecret( const std::string & encoding_type ) const GAuth::Secret GAuth::SecretsFile::clientSecret( const std::string & type ) const
{ {
reread() ; reread() ;
auto p = m_contents.m_map.find( clientKey(encoding_type) ) ; auto p = m_contents.m_map.find( clientKey(type) ) ;
if( p == m_contents.m_map.end() ) if( p == m_contents.m_map.end() )
{ {
return Secret::none() ; return Secret::none() ;
@ -258,25 +272,25 @@ GAuth::Secret GAuth::SecretsFile::clientSecret( const std::string & encoding_typ
{ {
std::string id_xtext = G::Str::head( (*p).second.s , " " ) ; std::string id_xtext = G::Str::head( (*p).second.s , " " ) ;
std::string secret_encoded = G::Str::tail( (*p).second.s , " " ) ; std::string secret_encoded = G::Str::tail( (*p).second.s , " " ) ;
return Secret( secret_encoded , canonical(encoding_type) , id_xtext , true , line((*p).second.n) ) ; return Secret( secret_encoded , canonical(type) , id_xtext , true , line((*p).second.n) ) ;
} }
} }
GAuth::Secret GAuth::SecretsFile::serverSecret( const std::string & encoding_type , const std::string & id ) const GAuth::Secret GAuth::SecretsFile::serverSecret( const std::string & type , const std::string & id ) const
{ {
if( id.empty() ) if( id.empty() )
return Secret::none() ; return Secret::none() ;
reread() ; reread() ;
auto p = m_contents.m_map.find( serverKey(encoding_type,G::Xtext::encode(id)) ) ; auto p = m_contents.m_map.find( serverKey(type,G::Xtext::encode(id)) ) ;
if( p == m_contents.m_map.end() ) if( p == m_contents.m_map.end() )
{ {
return Secret::none( id ) ; return Secret::none( id ) ;
} }
else else
{ {
return Secret( (*p).second.s , canonical(encoding_type) , id , false , line((*p).second.n) ) ; return Secret( (*p).second.s , canonical(type) , id , false , line((*p).second.n) ) ;
} }
} }
@ -285,9 +299,9 @@ std::pair<std::string,std::string> GAuth::SecretsFile::serverTrust( const std::s
reread() ; // (new) reread() ; // (new)
std::pair<std::string,std::string> result ; std::pair<std::string,std::string> result ;
std::string encoding_type = "none" ; std::string type = "none" ;
const std::string & id = address_range ; // the address-range lives in the id field const std::string & id = address_range ; // the address-range lives in the id field
auto p = m_contents.m_map.find( serverKey(encoding_type,G::Xtext::encode(id)) ) ; auto p = m_contents.m_map.find( serverKey(type,G::Xtext::encode(id)) ) ;
if( p != m_contents.m_map.end() ) if( p != m_contents.m_map.end() )
{ {
result.first = (*p).second.s ; // the trustee name lives in the shared-secret field result.first = (*p).second.s ; // the trustee name lives in the shared-secret field
@ -301,9 +315,11 @@ std::string GAuth::SecretsFile::path() const
return m_path.str() ; return m_path.str() ;
} }
bool GAuth::SecretsFile::contains( const std::string & encoding_type ) const bool GAuth::SecretsFile::contains( const std::string & type , const std::string & id ) const
{ {
return m_contents.m_types.find(canonical(encoding_type)) != m_contents.m_types.end() ; return id.empty() ?
m_contents.m_types.find( canonical(type) ) != m_contents.m_types.end() :
m_contents.m_map.find( serverKey(type,G::Xtext::encode(id)) ) != m_contents.m_map.end() ;
} }
std::string GAuth::SecretsFile::line( unsigned int line_number ) std::string GAuth::SecretsFile::line( unsigned int line_number )

View File

@ -51,21 +51,19 @@ public:
///< Checks the given file. Logs warnings and throws an exception ///< Checks the given file. Logs warnings and throws an exception
///< if there are any fatal errors. ///< if there are any fatal errors.
SecretsFile( const G::Path & path , bool auto_reread , const std::string & debug_name , SecretsFile( const G::Path & path , bool auto_reread , const std::string & debug_name ) ;
const std::string & server_type = std::string() ) ; ///< Constructor to read "client" and "server" records from
///< Constructor to read "client" and "<server-type>" records ///< the named file. The path is optional; see valid().
///< from the named file. The server type defaults to "server".
///< The filename path is optional; see valid().
bool valid() const ; bool valid() const ;
///< Returns true if the file path was supplied in the ctor. ///< Returns true if the file path was supplied in the ctor.
Secret clientSecret( const std::string & encoding_type ) const ; Secret clientSecret( const std::string & type ) const ;
///< Returns the client id and secret for the given encoding-type. ///< Returns the client id and secret for the given type.
///< Returns the empty string if none. ///< Returns the empty string if none.
Secret serverSecret( const std::string & encoding_type , const std::string & id ) const ; Secret serverSecret( const std::string & type , const std::string & id ) const ;
///< Returns the server secret for the given id and encoding-type. ///< Returns the server secret for the given id and type.
///< Returns the empty string if none. ///< Returns the empty string if none.
std::pair<std::string,std::string> serverTrust( const std::string & address_range ) const ; std::pair<std::string,std::string> serverTrust( const std::string & address_range ) const ;
@ -75,8 +73,9 @@ public:
std::string path() const ; std::string path() const ;
///< Returns the file path, as supplied to the ctor. ///< Returns the file path, as supplied to the ctor.
bool contains( const std::string & server_encoding_type ) const ; bool contains( const std::string & type , const std::string & id = {} ) const ;
///< Returns true if the given server encoding-type is represented. ///< Returns true if a server secret of the given type
///< is available for the particular user or any user.
private: private:
struct Value struct Value
@ -97,16 +96,17 @@ private:
} ; } ;
private: private:
void read( const G::Path & , bool ) ; void read( const G::Path & ) ;
void reread() const ; void reread() const ;
void reread( int ) ; void reread( int ) ;
static void checkImp( const std::string & path , bool strict , const std::string & server_type ) ; static Contents readContents( const G::Path & ) ;
static Contents readContents( const G::Path & , const std::string & , bool ) ; static Contents readContents( std::istream & ) ;
static Contents readContents( std::istream & , const std::string & , bool ) ; static void processLine( Contents & ,
static void processLine( Contents & , const std::string & server_type , unsigned int , const std::string & side , const std::string & , const std::string & , const std::string & ) ;
unsigned int , const std::string & side , const std::string & , const std::string & , const std::string & , bool ) ; static void processLineImp( Contents & ,
static void showWarnings( const Warnings & warnings , const G::Path & path , const std::string & debug_name = std::string() ) ; unsigned int , const std::string & side , const std::string & , const std::string & , const std::string & ) ;
static void addWarning( Contents & , unsigned int , const std::string & , const std::string & ) ; static void showWarnings( const Warnings & warnings , const G::Path & path , const std::string & debug_name = {} ) ;
static void addWarning( Contents & , unsigned int , const std::string & , const std::string & = {} ) ;
static std::string canonical( const std::string & encoding_type ) ; static std::string canonical( const std::string & encoding_type ) ;
static std::string serverKey( const std::string & , const std::string & ) ; static std::string serverKey( const std::string & , const std::string & ) ;
static std::string clientKey( const std::string & ) ; static std::string clientKey( const std::string & ) ;
@ -117,7 +117,6 @@ private:
G::Path m_path ; G::Path m_path ;
bool m_auto ; bool m_auto ;
std::string m_debug_name ; std::string m_debug_name ;
std::string m_server_type ;
bool m_valid ; bool m_valid ;
Contents m_contents ; Contents m_contents ;
G::SystemTime m_file_time ; G::SystemTime m_file_time ;

View File

@ -3,9 +3,6 @@
/* Define true to use epoll */ /* Define true to use epoll */
#undef GCONFIG_ENABLE_EPOLL #undef GCONFIG_ENABLE_EPOLL
/* Define true to use IPv6 */
#undef GCONFIG_ENABLE_IPV6
/* Define true to use std::thread */ /* Define true to use std::thread */
#undef GCONFIG_ENABLE_STD_THREAD #undef GCONFIG_ENABLE_STD_THREAD

View File

@ -121,7 +121,7 @@ std::size_t G::Arg::match( const std::string & prefix ) const
{ {
for( std::size_t i = 1U ; i < m_array.size() ; i++ ) for( std::size_t i = 1U ; i < m_array.size() ; i++ )
{ {
if( G::Str::headMatch(m_array[i],prefix) ) if( Str::headMatch(m_array[i],prefix) )
{ {
return i ; return i ;
} }
@ -205,8 +205,8 @@ void G::Arg::parseImp( const std::string & command_line )
string_view nbws( "\0\0" , 2U ) ; string_view nbws( "\0\0" , 2U ) ;
const char esc = '\\' ; const char esc = '\\' ;
const char qq = '\"' ; const char qq = '\"' ;
G::Str::splitIntoTokens( G::Str::dequote(command_line,qq,esc,ws,nbws) , m_array , ws , esc ) ; Str::splitIntoTokens( Str::dequote(command_line,qq,esc,ws,nbws) , m_array , ws , esc ) ;
G::Str::replace( m_array , '\0' , ' ' ) ; Str::replace( m_array , '\0' , ' ' ) ;
} }
std::string G::Arg::exe( bool do_throw ) std::string G::Arg::exe( bool do_throw )

View File

@ -112,7 +112,7 @@ void G::Base64Imp::encode_imp( iterator_out result_p , string_in input , string_
auto const end = input.end() ; auto const end = input.end() ;
for( auto p = input.begin() ; p != end ; blocks++ ) for( auto p = input.begin() ; p != end ; blocks++ )
{ {
if( !eol.empty() && blocks && (blocks % blocks_per_line) == 0U ) if( !eol.empty() && blocks != 0U && (blocks % blocks_per_line) == 0U )
std::copy( eol.begin() , eol.end() , result_p ) ; std::copy( eol.begin() , eol.end() , result_p ) ;
uint32_type n = 0UL ; uint32_type n = 0UL ;
@ -252,7 +252,7 @@ void G::Base64Imp::generate_8( g_uint32_t & n , std::size_t & bits , iterator_ou
*result++ = to_char(hi_8(n)) ; *result++ = to_char(hi_8(n)) ;
n <<= 8U ; n <<= 8U ;
} }
else if( hi_8(n) ) else if( hi_8(n) != 0U )
{ {
error = true ; error = true ;
} }
@ -261,7 +261,7 @@ void G::Base64Imp::generate_8( g_uint32_t & n , std::size_t & bits , iterator_ou
std::size_t G::Base64Imp::index( char c , bool & error ) noexcept std::size_t G::Base64Imp::index( char c , bool & error ) noexcept
{ {
std::size_t pos = character_map.find( c ) ; std::size_t pos = character_map.find( c ) ;
error = error || !c || pos == std::string::npos ; error = error || (c=='\0') || pos == std::string::npos ;
return pos == std::string::npos ? std::size_t(0) : pos ; return pos == std::string::npos ? std::size_t(0) : pos ;
} }

View File

@ -44,10 +44,10 @@ class G::BatchFile
public: public:
G_EXCEPTION( Error , "batch file error" ) ; G_EXCEPTION( Error , "batch file error" ) ;
explicit BatchFile( const G::Path & ) ; explicit BatchFile( const Path & ) ;
///< Constructor that reads from a file. ///< Constructor that reads from a file.
BatchFile( const G::Path & , std::nothrow_t ) ; BatchFile( const Path & , std::nothrow_t ) ;
///< Constructor that reads from a file that might be missing ///< Constructor that reads from a file that might be missing
///< or empty. ///< or empty.
@ -61,14 +61,14 @@ public:
std::string name() const ; std::string name() const ;
///< Returns the "start" window name, if any. ///< Returns the "start" window name, if any.
const G::StringArray & args() const ; const StringArray & args() const ;
///< Returns the startup command-line broken up into de-quoted pieces. ///< Returns the startup command-line broken up into de-quoted pieces.
///< The first item in the list will be the executable. ///< The first item in the list will be the executable.
std::size_t lineArgsPos() const ; std::size_t lineArgsPos() const ;
///< Returns the position in line() where the arguments start. ///< Returns the position in line() where the arguments start.
static void write( const G::Path & , const StringArray & args , static void write( const Path & , const StringArray & args ,
const std::string & start_window_name = std::string() ) ; const std::string & start_window_name = std::string() ) ;
///< Writes a startup batch file, including a "start" prefix. ///< Writes a startup batch file, including a "start" prefix.
///< If the "start" window name is not supplied then it is ///< If the "start" window name is not supplied then it is
@ -87,7 +87,7 @@ private:
private: private:
std::string m_line ; std::string m_line ;
std::string m_name ; std::string m_name ;
G::StringArray m_args ; StringArray m_args ;
} ; } ;
#endif #endif

View File

@ -87,10 +87,14 @@ namespace G
const T & at( std::size_t i ) const { checkindex( i ) ; return *(m_p+i) ; } const T & at( std::size_t i ) const { checkindex( i ) ; return *(m_p+i) ; }
T & at( std::size_t i ) { checkindex( i ) ; return *(m_p+i) ; } T & at( std::size_t i ) { checkindex( i ) ; return *(m_p+i) ; }
std::size_t size() const noexcept { return m_n ; } std::size_t size() const noexcept { return m_n ; }
std::size_t capacity() const noexcept { return m_c ; }
bool empty() const noexcept { return m_n == 0U ; } bool empty() const noexcept { return m_n == 0U ; }
void clear() noexcept { m_n = 0 ; } void clear() noexcept { m_n = 0 ; }
void shrink_to_fit() noexcept { if( empty() && m_p ) { std::free(m_p) ; m_p = nullptr ; m_c = 0U ; } } // NOLINT cppcoreguidelines-no-malloc
iterator begin() noexcept { return m_p ? m_p : &m_c0 ; } iterator begin() noexcept { return m_p ? m_p : &m_c0 ; }
iterator end() noexcept { return m_p ? (m_p+m_n) : &m_c0 ; } iterator end() noexcept { return m_p ? (m_p+m_n) : &m_c0 ; }
const value_type * data() const noexcept { return m_p ? m_p : &m_c0 ; }
value_type * data() noexcept { return m_p ? m_p : &m_c0 ; }
const_iterator begin() const noexcept { return m_p ? m_p : &m_c0 ; } const_iterator begin() const noexcept { return m_p ? m_p : &m_c0 ; }
const_iterator end() const noexcept { return m_p ? (m_p+m_n) : &m_c0 ; } const_iterator end() const noexcept { return m_p ? (m_p+m_n) : &m_c0 ; }
const_iterator cbegin() const noexcept { return m_p ? m_p : &m_c0 ; } const_iterator cbegin() const noexcept { return m_p ? m_p : &m_c0 ; }
@ -182,7 +186,7 @@ namespace G
char * m_p{nullptr} ; char * m_p{nullptr} ;
std::size_t m_n{0U} ; std::size_t m_n{0U} ;
std::size_t m_c{0U} ; std::size_t m_c{0U} ;
char m_c0{'\0'} ; value_type m_c0{'\0'} ;
} ; } ;
template <typename Uptr, typename T = char> template <typename Uptr, typename T = char>
@ -193,7 +197,7 @@ namespace G
G_ASSERT( p == nullptr || p == &buffer[0] ) ; // assert malloc is behaving G_ASSERT( p == nullptr || p == &buffer[0] ) ; // assert malloc is behaving
if( p != &buffer[0] ) if( p != &buffer[0] )
throw std::bad_cast() ; // buffer too small for a U throw std::bad_cast() ; // buffer too small for a U
return new(p) U ; // placement new return new(p) U ;
} }
template <typename Uptr, typename T = char> template <typename Uptr, typename T = char>
@ -204,7 +208,7 @@ namespace G
G_ASSERT( p == nullptr || p == &buffer[0] ) ; // assert malloc is behaving G_ASSERT( p == nullptr || p == &buffer[0] ) ; // assert malloc is behaving
if( p != &buffer[0] ) if( p != &buffer[0] )
return nullptr ; // buffer too small for a U return nullptr ; // buffer too small for a U
return new(p) U ; // placement new return new(p) U ;
} }
template <typename Uptr, typename T = char> template <typename Uptr, typename T = char>
@ -214,7 +218,7 @@ namespace G
return const_cast<Uptr>( buffer_cast<U*>( const_cast<Buffer<T>&>(buffer) ) ) ; return const_cast<Uptr>( buffer_cast<U*>( const_cast<Buffer<T>&>(buffer) ) ) ;
} }
template <typename T> void swap( Buffer<T> & a , Buffer<T> & b ) template <typename T> void swap( Buffer<T> & a , Buffer<T> & b ) noexcept
{ {
a.swap( b ) ; a.swap( b ) ;
} }

View File

@ -186,9 +186,9 @@ void G::CleanupImp::installHandler( int signum )
bool G::CleanupImp::ignored( int signum ) bool G::CleanupImp::ignored( int signum )
{ {
struct ::sigaction action {} ; struct ::sigaction action {} ;
if( ::sigaction( signum , nullptr , &action ) ) if( ::sigaction( signum , nullptr , &action ) != 0 )
throw Cleanup::Error( "sigaction" ) ; throw Cleanup::Error( "sigaction" ) ;
return action.sa_handler == SIG_IGN ; return action.sa_handler == SIG_IGN ; // NOLINT
} }
void G::CleanupImp::installDefault( int signum ) void G::CleanupImp::installDefault( int signum )
@ -203,7 +203,7 @@ void G::CleanupImp::installDefault( const G::SignalSafe & , int signum )
void G::CleanupImp::installIgnore( int signum ) void G::CleanupImp::installIgnore( int signum )
{ {
install( signum , SIG_IGN , true ) ; install( signum , SIG_IGN , true ) ; // NOLINT
} }
void G::CleanupImp::install( int signum , Handler fn , bool do_throw ) void G::CleanupImp::install( int signum , Handler fn , bool do_throw )
@ -211,7 +211,7 @@ void G::CleanupImp::install( int signum , Handler fn , bool do_throw )
// install the given handler, or the system default if null // install the given handler, or the system default if null
struct ::sigaction action {} ; struct ::sigaction action {} ;
action.sa_handler = fn ; action.sa_handler = fn ;
if( ::sigaction( signum , &action , nullptr ) && do_throw ) if( ::sigaction( signum , &action , nullptr ) != 0 && do_throw )
throw Cleanup::Error( "sigaction" ) ; throw Cleanup::Error( "sigaction" ) ;
} }

View File

@ -57,7 +57,8 @@ namespace G
} }
} }
G::BrokenDownTime::BrokenDownTime() G::BrokenDownTime::BrokenDownTime() :
m_tm{}
{ {
m_tm.tm_isdst = -1 ; m_tm.tm_isdst = -1 ;
} }
@ -97,7 +98,7 @@ std::time_t G::BrokenDownTime::epochTimeFromUtcImp() const
std::time_t G::BrokenDownTime::epochTimeFromUtcImp( bool & diff_set , std::time_t & diff ) const std::time_t G::BrokenDownTime::epochTimeFromUtcImp( bool & diff_set , std::time_t & diff ) const
{ {
constexpr int day = 24 * 60 * 60 ; constexpr int day = 24 * 60 * 60 ;
constexpr std::time_t dt = 60 * 15 ; // india 30mins, nepal 45mins constexpr std::time_t dt = 60 * 15 ; // india 30mins, nepal 45mins // NOLINT
constexpr std::time_t day_and_a_bit = day + dt ; constexpr std::time_t day_and_a_bit = day + dt ;
constexpr std::time_t t_rounding = 30 ; constexpr std::time_t t_rounding = 30 ;
@ -136,7 +137,7 @@ G::BrokenDownTime G::BrokenDownTime::local( SystemTime t )
{ {
BrokenDownTime tm ; BrokenDownTime tm ;
std::time_t s = t.s() ; std::time_t s = t.s() ;
if( ! localtime_r( &s , &tm.m_tm ) ) if( localtime_r( &s , &tm.m_tm ) == nullptr )
throw DateTime::Error() ; throw DateTime::Error() ;
return tm ; return tm ;
} }
@ -145,7 +146,7 @@ G::BrokenDownTime G::BrokenDownTime::utc( SystemTime t )
{ {
BrokenDownTime tm ; BrokenDownTime tm ;
std::time_t s = t.s() ; std::time_t s = t.s() ;
if( ! gmtime_r( &s , &tm.m_tm ) ) if( gmtime_r( &s , &tm.m_tm ) == nullptr )
throw DateTime::Error() ; throw DateTime::Error() ;
return tm ; return tm ;
} }
@ -364,7 +365,7 @@ void G::SystemTime::operator+=( TimeInterval i )
void G::SystemTime::streamOut( std::ostream & stream ) const void G::SystemTime::streamOut( std::ostream & stream ) const
{ {
std::streamsize w = stream.width() ; int w = static_cast<int>( stream.width() ) ;
char c = stream.fill() ; char c = stream.fill() ;
stream stream
<< s() << "." << s() << "."
@ -630,7 +631,7 @@ void G::TimeInterval::operator-=( TimeInterval i )
void G::TimeInterval::streamOut( std::ostream & stream ) const void G::TimeInterval::streamOut( std::ostream & stream ) const
{ {
std::streamsize w = stream.width() ; int w = static_cast<int>( stream.width() ) ;
char c = stream.fill() ; char c = stream.fill() ;
stream stream
<< s() << "." << s() << "."

View File

@ -1363,7 +1363,7 @@
#endif #endif
#if ! GCONFIG_HAVE_CXX_ALIGNMENT #if ! GCONFIG_HAVE_CXX_ALIGNMENT
namespace std namespace std // NOLINT
{ {
// missing in gcc 4.8.4 -- original copyright 2001-2016 FSF Inc, GPLv3 // missing in gcc 4.8.4 -- original copyright 2001-2016 FSF Inc, GPLv3
inline void * align( size_t align , size_t size , void * & ptr_inout , size_t & space ) noexcept inline void * align( size_t align , size_t size , void * & ptr_inout , size_t & space ) noexcept

View File

@ -100,18 +100,3 @@ G::Exception::Exception( const std::string & what , const std::string & more1 ,
{ {
} }
G::Exception G::Exception::translated() const
{
try
{
std::string head = G::Str::head( what() , ": " , false ) ;
std::string tail = G::Str::tail( what() , ": " ) ;
std::string new_head = G::gettext( head.c_str() ) ;
return G::Exception( new_head , tail ) ;
}
catch( std::exception & )
{
return *this ;
}
}

View File

@ -67,11 +67,6 @@ public:
Exception( const std::string & what , const std::string & more1 , const std::string & more2 , const std::string & more3 ) ; Exception( const std::string & what , const std::string & more1 , const std::string & more2 , const std::string & more3 ) ;
///< Constructor. ///< Constructor.
Exception translated() const ;
///< Returns a new exception that is possibly translated using
///< gettext(). (Use gettext_noop() in G_EXCEPTION declarations
///< to mark the string for translation.)
} ; } ;
#define G_EXCEPTION_CLASS( class_name , description ) class class_name : public G::Exception { public: class_name() : G::Exception((description)) {} explicit class_name(const char *more) : G::Exception((description),more) {} explicit class_name(const std::string &more) : G::Exception((description),more) {} class_name(const std::string &more1,const std::string &more2) : G::Exception((description),more1,more2) {} class_name(const std::string &more1,const std::string &more2,const std::string &more3) : G::Exception((description),more1,more2,more3) {} } #define G_EXCEPTION_CLASS( class_name , description ) class class_name : public G::Exception { public: class_name() : G::Exception((description)) {} explicit class_name(const char *more) : G::Exception((description),more) {} explicit class_name(const std::string &more) : G::Exception((description),more) {} class_name(const std::string &more1,const std::string &more2) : G::Exception((description),more1,more2) {} class_name(const std::string &more1,const std::string &more2,const std::string &more3) : G::Exception((description),more1,more2,more3) {} }

View File

@ -274,13 +274,14 @@ bool G::File::mkdirsr( int * ep , const Path & path , int limit )
bool G::File::mkdirs( const Path & path , std::nothrow_t , int limit ) bool G::File::mkdirs( const Path & path , std::nothrow_t , int limit )
{ {
int e = 0 ; int e = 0 ;
return mkdirsr( &e , path , limit ) ; bool ok = mkdirsr( &e , path , limit ) ;
return ok || e == EEXIST ;
} }
void G::File::mkdirs( const Path & path , int limit ) void G::File::mkdirs( const Path & path , int limit )
{ {
int e = 0 ; int e = 0 ;
if( !mkdirsr(&e,path,limit) ) if( !mkdirsr(&e,path,limit) && e != EEXIST )
throw CannotMkdir( path.str() , e ? G::Process::strerror(e) : std::string() ) ; throw CannotMkdir( path.str() , e ? G::Process::strerror(e) : std::string() ) ;
} }

View File

@ -93,7 +93,7 @@ std::filebuf * G::File::open( std::filebuf & fb , const Path & path , InOut inou
inout == InOut::In ? inout == InOut::In ?
FileImp::open( fb , path.cstr() , std::ios_base::in | std::ios_base::binary ) : FileImp::open( fb , path.cstr() , std::ios_base::in | std::ios_base::binary ) :
FileImp::open( fb , path.cstr() , std::ios_base::out | std::ios_base::binary ) ; FileImp::open( fb , path.cstr() , std::ios_base::out | std::ios_base::binary ) ;
return &fb ; return fb.is_open() ? &fb : nullptr ;
} }
int G::File::open( const char * path , InOutAppend mode ) noexcept int G::File::open( const char * path , InOutAppend mode ) noexcept
@ -146,12 +146,16 @@ void G::File::close( int fd ) noexcept
int G::File::mkdirImp( const Path & dir ) noexcept int G::File::mkdirImp( const Path & dir ) noexcept
{ {
int rc = _mkdir( dir.cstr() ) ; int rc = _mkdir( dir.cstr() ) ;
if( rc != 0 ) if( rc == 0 )
{ {
rc = G::Process::errno_() ; return 0 ;
if( rc == 0 ) rc = EINVAL ; }
else
{
int e = G::Process::errno_() ;
if( e == 0 ) e = EINVAL ;
return e ;
} }
return rc ;
} }
G::File::Stat G::File::statImp( const char * path , bool ) noexcept G::File::Stat G::File::statImp( const char * path , bool ) noexcept

View File

@ -38,7 +38,7 @@ class G::HashStateImp
{ {
public: public:
template <typename U> static std::string extension( U n ) ; template <typename U> static std::string extension( U n ) ;
///< Returns the given data size a four-character ///< Returns the given data size as a four-character
///< string. ///< string.
protected: protected:
@ -115,7 +115,7 @@ std::string G::HashState<N,U,S>::encode( const uint_type * values )
std::string result( N , '\0' ) ; std::string result( N , '\0' ) ;
for( std::size_t i = 0U ; i < N/4 ; i++ ) for( std::size_t i = 0U ; i < N/4 ; i++ )
{ {
convert_( values[i] , result.begin() + (i*4U) ) ; convert_( values[i] , result.begin() + (i*4U) ) ; // NOLINT narrowing
} }
return result ; return result ;
} }
@ -126,7 +126,7 @@ std::string G::HashState<N,U,S>::encode( const uint_type * values , size_type n
std::string result( N+4U , '\0' ) ; std::string result( N+4U , '\0' ) ;
for( std::size_t i = 0U ; i < N/4 ; i++ ) for( std::size_t i = 0U ; i < N/4 ; i++ )
{ {
convert_( values[i] , result.begin() + (i*4U) ) ; convert_( values[i] , result.begin() + (i*4U) ) ; // NOLINT narrowing
} }
convert_( n , result.begin() + N ) ; convert_( n , result.begin() + N ) ;
return result ; return result ;
@ -229,7 +229,7 @@ void G::HashState<N,U,S>::convert( const std::string & str , uint_type & n_out )
template <unsigned int N, typename U, typename S> template <unsigned int N, typename U, typename S>
void G::HashState<N,U,S>::convert( const std::string & str , uint_type * state_out ) void G::HashState<N,U,S>::convert( const std::string & str , uint_type * state_out )
{ {
for( unsigned int i = 0U ; i < (N/4U) ; i++ ) for( std::size_t i = 0U ; i < (N/4U) ; i++ )
{ {
convert( str.at(i*4U+3U) , str.at(i*4U+2U) , str.at(i*4U+1U) , str.at(i*4U+0U) , state_out[i] ) ; convert( str.at(i*4U+3U) , str.at(i*4U+2U) , str.at(i*4U+1U) , str.at(i*4U+0U) , state_out[i] ) ;
} }

View File

@ -98,12 +98,12 @@ bool G::Identity::isRoot() const noexcept
return false ; return false ;
} }
bool G::Identity::operator==( const Identity & other ) const noexcept bool G::Identity::operator==( const Identity & ) const noexcept
{ {
return true ; return true ;
} }
bool G::Identity::operator!=( const Identity & other ) const noexcept bool G::Identity::operator!=( const Identity & ) const noexcept
{ {
return false ; return false ;
} }

View File

@ -209,6 +209,7 @@ std::ostream & G::LogOutput::start( Log::Severity severity )
open( m_path , false ) ; open( m_path , false ) ;
std::ostream & ss = LogOutputImp::ostream1() ; std::ostream & ss = LogOutputImp::ostream1() ;
ss << std::dec ;
if( m_exename.length() ) if( m_exename.length() )
ss << m_exename << ": " ; ss << m_exename << ": " ;
if( m_config.m_with_timestamp ) if( m_config.m_with_timestamp )

View File

@ -298,8 +298,8 @@ G::Md5Imp::big_t G::Md5Imp::digest::op( const block & m , aux_fn_t aux , big_t a
G::Md5Imp::big_t G::Md5Imp::digest::rot32( small_t places , big_t n ) G::Md5Imp::big_t G::Md5Imp::digest::rot32( small_t places , big_t n )
{ {
// circular rotate of 32 LSBs, with corruption of higher bits // circular rotate of 32 LSBs, with corruption of higher bits
big_t overflow_mask = ( 1UL << places ) - 1UL ; // in case big_t is more than 32 bits big_t overflow_mask = ( big_t(1U) << places ) - big_t(1U) ; // in case big_t is more than 32 bits
big_t overflow = ( n >> ( 32U - places ) ) ; big_t overflow = ( n >> ( small_t(32U) - places ) ) ;
return ( n << places ) | ( overflow & overflow_mask ) ; return ( n << places ) | ( overflow & overflow_mask ) ;
} }

View File

@ -29,6 +29,8 @@
#include "gprocess.h" #include "gprocess.h"
#include "gpath.h" #include "gpath.h"
#include "gstrings.h" #include "gstrings.h"
#include <memory>
#include <new>
#include <string> #include <string>
#include <future> #include <future>
@ -207,7 +209,7 @@ public:
///< \code ///< \code
///< std::promise<int,std::string> p ; ///< std::promise<int,std::string> p ;
///< std::future<int,std::string> f = p.get_future() ; ///< std::future<int,std::string> f = p.get_future() ;
///< std::thread t( std::bind(&NewPromiseWaitable::waitp,waitable,_1) , std::move(p) ) ; ///< std::thread t( std::bind(&NewProcessWaitable::waitp,waitable,_1) , std::move(p) ) ;
///< f.wait() ; ///< f.wait() ;
///< t.join() ; ///< t.join() ;
///< int e = f.get() ; ///< int e = f.get() ;

View File

@ -142,7 +142,7 @@ void G::NewProcess::kill( bool yield ) noexcept
if( yield ) if( yield )
{ {
G::threading::yield() ; G::threading::yield() ;
::close( ::open( "/dev/null" , O_RDONLY ) ) ; // hmm ::close( ::open( "/dev/null" , O_RDONLY ) ) ; // hmm // NOLINT
G::threading::yield() ; G::threading::yield() ;
} }
} }
@ -322,7 +322,7 @@ bool G::NewProcessImp::duplicate( Fd fd , int fd_std )
G_ASSERT( !(fd==Fd::pipe()) ) ; G_ASSERT( !(fd==Fd::pipe()) ) ;
if( fd == Fd::devnull() ) if( fd == Fd::devnull() )
{ {
int fd_null = ::open( G::Path::nullDevice().cstr() , fd_std == STDIN_FILENO ? O_RDONLY : O_WRONLY ) ; int fd_null = ::open( G::Path::nullDevice().cstr() , fd_std == STDIN_FILENO ? O_RDONLY : O_WRONLY ) ; // NOLINT
if( fd_null < 0 ) throw NewProcess::Error( "failed to open /dev/null" ) ; if( fd_null < 0 ) throw NewProcess::Error( "failed to open /dev/null" ) ;
::dup2( fd_null , fd_std ) ; ::dup2( fd_null , fd_std ) ;
return true ; return true ;
@ -429,7 +429,7 @@ G::NewProcessWaitable & G::NewProcessWaitable::wait()
{ {
// (worker thread - keep it simple - never throws - does read then waitpid) // (worker thread - keep it simple - never throws - does read then waitpid)
{ {
std::array<char,64U> discard ; // NOLINT cppcoreguidelines-pro-type-member-init std::array<char,64U> discard {} ;
char * p = &m_buffer[0] ; char * p = &m_buffer[0] ;
std::size_t space = m_buffer.size() ; std::size_t space = m_buffer.size() ;
std::size_t size = 0U ; std::size_t size = 0U ;
@ -469,7 +469,7 @@ G::NewProcessWaitable & G::NewProcessWaitable::wait()
errno = 0 ; errno = 0 ;
m_rc = ::waitpid( m_pid , &m_status , 0 ) ; m_rc = ::waitpid( m_pid , &m_status , 0 ) ;
m_error = errno ; m_error = errno ;
if( m_rc >= 0 && ( WIFSTOPPED(m_status) || WIFCONTINUED(m_status) ) ) if( m_rc >= 0 && ( WIFSTOPPED(m_status) || WIFCONTINUED(m_status) ) ) // NOLINT
; // keep waiting ; // keep waiting
else if( m_rc == -1 && m_error == EINTR ) else if( m_rc == -1 && m_error == EINTR )
; // keep waiting ; // keep waiting
@ -498,18 +498,18 @@ int G::NewProcessWaitable::get() const
ss << "errno=" << (m_read_error?m_read_error:m_error) ; ss << "errno=" << (m_read_error?m_read_error:m_error) ;
throw NewProcess::WaitError( ss.str() ) ; throw NewProcess::WaitError( ss.str() ) ;
} }
else if( !WIFEXITED(m_status) ) else if( !WIFEXITED(m_status) ) // NOLINT
{ {
// uncaught signal // uncaught signal
std::ostringstream ss ; std::ostringstream ss ;
ss << "pid=" << m_pid ; ss << "pid=" << m_pid ;
if( WIFSIGNALED(m_status) ) if( WIFSIGNALED(m_status) ) // NOLINT
ss << " signal=" << WTERMSIG(m_status) ; ss << " signal=" << WTERMSIG(m_status) ; // NOLINT
throw NewProcess::ChildError( ss.str() ) ; throw NewProcess::ChildError( ss.str() ) ;
} }
else else
{ {
result = WEXITSTATUS( m_status ) ; result = WEXITSTATUS( m_status ) ; // NOLINT
} }
} }
return result ; return result ;

View File

@ -63,7 +63,7 @@ public:
const std::string in_type ; // "password", "prompt", "error", "info" const std::string in_type ; // "password", "prompt", "error", "info"
const std::string in ; // password prompt, non-password prompt, error text, infomation message, etc const std::string in ; // password prompt, non-password prompt, error text, infomation message, etc
std::string out ; // password, or whatever was prompted for std::string out ; // password, or whatever was prompted for
bool out_defined ; // to be set to true if 'out' is assigned bool out_defined{false} ; // to be set to true if 'out' is assigned
} ; } ;
class Error : public G::Exception /// An exception class for G::Pam. class Error : public G::Exception /// An exception class for G::Pam.
{ {

View File

@ -54,24 +54,8 @@ class G::PamImp
{ {
public: public:
using Handle = pam_handle_t * ; using Handle = pam_handle_t * ;
private:
using Conversation = struct pam_conv ; using Conversation = struct pam_conv ;
using Error = Pam::Error ;
using ItemArray = Pam::ItemArray ;
static constexpr int MAGIC = 3456 ;
public:
Pam & m_pam ;
int m_magic ;
mutable int m_rc ; // required for pam_end()
Handle m_hpam ;
Conversation m_conv ;
bool m_silent ;
public:
PamImp( Pam & pam , const std::string & app , const std::string & user , bool silent ) ; PamImp( Pam & pam , const std::string & app , const std::string & user , bool silent ) ;
~PamImp() ;
Handle hpam() const ; Handle hpam() const ;
bool silent() const ; bool silent() const ;
bool authenticate( bool ) ; bool authenticate( bool ) ;
@ -84,8 +68,24 @@ public:
std::string name() const ; std::string name() const ;
public: public:
~PamImp() ;
PamImp( const PamImp & ) = delete ; PamImp( const PamImp & ) = delete ;
void operator=( const PamImp & ) = delete ; PamImp( PamImp && ) = delete ;
PamImp & operator=( const PamImp & ) = delete ;
PamImp & operator=( PamImp && ) = delete ;
public:
Pam & m_pam ;
int m_magic ;
mutable int m_rc ; // required for pam_end()
Handle m_hpam ;
Conversation m_conv ;
bool m_silent ;
private:
using Error = Pam::Error ;
using ItemArray = Pam::ItemArray ;
static constexpr int MAGIC = 3456 ;
private: private:
static int converseCallback( int n , G_PAM_CONST struct pam_message ** in , static int converseCallback( int n , G_PAM_CONST struct pam_message ** in ,
@ -102,6 +102,8 @@ G::PamImp::PamImp( G::Pam & pam , const std::string & application , const std::s
m_pam(pam) , m_pam(pam) ,
m_magic(MAGIC) , m_magic(MAGIC) ,
m_rc(PAM_SUCCESS) , m_rc(PAM_SUCCESS) ,
m_hpam(nullptr) ,
m_conv{} ,
m_silent(silent) m_silent(silent)
{ {
G_DEBUG( "G::PamImp::ctor: [" << application << "] [" << user << "]" ) ; G_DEBUG( "G::PamImp::ctor: [" << application << "] [" << user << "]" ) ;
@ -207,10 +209,10 @@ void G::PamImp::release( struct pam_response * rsp , std::size_t n )
for( std::size_t i = 0U ; i < n ; i++ ) for( std::size_t i = 0U ; i < n ; i++ )
{ {
if( rsp[i].resp != nullptr ) if( rsp[i].resp != nullptr )
std::free( rsp[i].resp ) ; std::free( rsp[i].resp ) ; // NOLINT
} }
} }
std::free( rsp ) ; std::free( rsp ) ; // NOLINT
} }
int G::PamImp::converseCallback( int n_in , G_PAM_CONST struct pam_message ** in , int G::PamImp::converseCallback( int n_in , G_PAM_CONST struct pam_message ** in ,
@ -266,7 +268,7 @@ int G::PamImp::converseCallback( int n_in , G_PAM_CONST struct pam_message ** in
// allocate the response - treat "out" as a pointer to a pointer // allocate the response - treat "out" as a pointer to a pointer
// to a contiguous array of structures (see linux man pam_conv) // to a contiguous array of structures (see linux man pam_conv)
// //
rsp = static_cast<struct pam_response*>( std::malloc(n*sizeof(struct pam_response)) ) ; rsp = static_cast<struct pam_response*>( std::malloc(n*sizeof(struct pam_response)) ) ; // NOLINT
if( rsp == nullptr ) if( rsp == nullptr )
throw std::bad_alloc() ; throw std::bad_alloc() ;
for( std::size_t j = 0U ; j < n ; j++ ) for( std::size_t j = 0U ; j < n ; j++ )
@ -349,7 +351,7 @@ void G::PamImp::check( const std::string & op , int rc ) const
char * G::PamImp::strdup_( const char * p ) char * G::PamImp::strdup_( const char * p )
{ {
p = p ? p : "" ; p = p ? p : "" ;
char * copy = static_cast<char*>( std::malloc(std::strlen(p)+1U) ) ; char * copy = static_cast<char*>( std::malloc(std::strlen(p)+1U) ) ; // NOLINT
if( copy != nullptr ) if( copy != nullptr )
std::strcpy( copy , p ) ; // NOLINT std::strcpy( copy , p ) ; // NOLINT
return copy ; return copy ;

View File

@ -140,7 +140,7 @@ public:
///< Returns !isRelative(). ///< Returns !isRelative().
bool isRelative() const ; bool isRelative() const ;
///< Returns true if the path is a relative path. ///< Returns true if the path is a relative path or empty().
void pathAppend( const std::string & tail ) ; void pathAppend( const std::string & tail ) ;
///< Appends a filename or a relative path to this path. ///< Appends a filename or a relative path to this path.

View File

@ -235,9 +235,9 @@ std::string G::Process::exe()
// == // ==
G::Process::Id::Id() noexcept G::Process::Id::Id() noexcept :
m_pid(::getpid())
{ {
m_pid = ::getpid() ;
} }
std::string G::Process::Id::str() const std::string G::Process::Id::str() const

View File

@ -54,12 +54,14 @@ G::Root::Root( bool change_group ) :
} }
} }
G::Root::~Root() G::Root::~Root() // NOLINT bugprone-exception-escape
{ {
if( m_this == this && m_initialised ) if( m_this == this && m_initialised )
{ {
m_this = nullptr ; m_this = nullptr ;
Process::beOrdinary( m_nobody , m_change_group ) ; // std::terminate on error int e_saved = Process::errno_() ;
Process::beOrdinary( m_nobody , m_change_group ) ; // can throw - std::terminate is correct
Process::errno_( SignalSafe() , e_saved ) ;
} }
} }

View File

@ -64,7 +64,7 @@ public:
///< Constructor overload with explicit control over whether to change the ///< Constructor overload with explicit control over whether to change the
///< group-id or not. ///< group-id or not.
~Root() ; ~Root() ; // NOLINT
///< Desctructor. Releases special privileges if this instance acquired them. ///< Desctructor. Releases special privileges if this instance acquired them.
///< The implementation uses G::Process::beOrdinary(). Errors from seteuid() ///< The implementation uses G::Process::beOrdinary(). Errors from seteuid()
///< will call Process::terminate(). ///< will call Process::terminate().

View File

@ -666,7 +666,7 @@ unsigned long G::StrImp::toULongHex( const std::string & s , bool limited )
else if( c >= 65U && c <= 70U ) c -= 55U ; else if( c >= 65U && c <= 70U ) c -= 55U ;
else if( c >= 48U && c <= 57U ) c -= 48U ; else if( c >= 48U && c <= 57U ) c -= 48U ;
else throw Str::InvalidFormat( "invalid hexadecimal" , s ) ; else throw Str::InvalidFormat( "invalid hexadecimal" , s ) ;
n <<= 4 ; n <<= 4U ;
n += c ; n += c ;
} }
return n ; return n ;
@ -764,8 +764,8 @@ std::size_t G::StrImp::outputHex( Tout out , char c )
namespace imp = G::StrImp ; namespace imp = G::StrImp ;
std::size_t n = static_cast<unsigned char>( c ) ; std::size_t n = static_cast<unsigned char>( c ) ;
n &= 0xFFU ; n &= 0xFFU ;
*out++ = imp::chars_hexmap[(n>>4)%16U] ; *out++ = imp::chars_hexmap[(n>>4U)%16U] ;
*out++ = imp::chars_hexmap[(n>>0)%16U] ; *out++ = imp::chars_hexmap[(n>>0U)%16U] ;
return 2U ; return 2U ;
} }
@ -776,10 +776,10 @@ std::size_t G::StrImp::outputHex( Tout out , wchar_t c )
using uwchar_t = typename std::make_unsigned<wchar_t>::type ; using uwchar_t = typename std::make_unsigned<wchar_t>::type ;
std::size_t n = static_cast<uwchar_t>( c ) ; std::size_t n = static_cast<uwchar_t>( c ) ;
n &= 0xFFFFU ; n &= 0xFFFFU ;
*out++ = imp::chars_hexmap[(n>>12)%16U] ; *out++ = imp::chars_hexmap[(n>>12U)%16U] ;
*out++ = imp::chars_hexmap[(n>>8)%16U] ; *out++ = imp::chars_hexmap[(n>>8U)%16U] ;
*out++ = imp::chars_hexmap[(n>>4)%16U] ; *out++ = imp::chars_hexmap[(n>>4U)%16U] ;
*out++ = imp::chars_hexmap[(n>>0)%16U] ; *out++ = imp::chars_hexmap[(n>>0U)%16U] ;
return 4U ; return 4U ;
} }
@ -1443,7 +1443,7 @@ std::size_t G::Str::ifind( const std::string & s , const std::string & key , std
{ {
namespace imp = G::StrImp ; namespace imp = G::StrImp ;
if( s.empty() || key.empty() || pos > s.length() ) return std::string::npos ; if( s.empty() || key.empty() || pos > s.length() ) return std::string::npos ;
auto p = std::search( s.begin()+pos , s.end() , key.begin() , key.end() , imp::imatchc ) ; auto p = std::search( s.begin()+pos , s.end() , key.begin() , key.end() , imp::imatchc ) ; // NOLINT narrowing
return p == s.end() ? std::string::npos : std::distance(s.begin(),p) ; return p == s.end() ? std::string::npos : std::distance(s.begin(),p) ;
} }

View File

@ -97,7 +97,7 @@ public:
} }
constexpr static bool same( basic_string_view a , basic_string_view b ) noexcept constexpr static bool same( basic_string_view a , basic_string_view b ) noexcept
{ {
return a.size() == b.size() && StringViewImp::same( a.size() , a.data() , b.data() ) ; return a.size() == b.size() && ( a.empty() || StringViewImp::same( a.size() , a.data() , b.data() ) ) ;
} }
constexpr std::size_t size() const noexcept { return m_n ; } constexpr std::size_t size() const noexcept { return m_n ; }
constexpr std::size_t length() const noexcept { return m_n ; } constexpr std::size_t length() const noexcept { return m_n ; }
@ -106,10 +106,10 @@ public:
void swap( basic_string_view<Tchar> & other ) noexcept { std::swap(m_p,other.m_p) ; std::swap(m_n,other.m_n) ; } void swap( basic_string_view<Tchar> & other ) noexcept { std::swap(m_p,other.m_p) ; std::swap(m_n,other.m_n) ; }
constexpr const Tchar & operator[]( std::size_t i ) const { return m_p[i] ; } constexpr const Tchar & operator[]( std::size_t i ) const { return m_p[i] ; }
const Tchar & at( std::size_t i ) const { if( i >= m_n ) throw std::out_of_range("string_view") ; return m_p[i] ; } const Tchar & at( std::size_t i ) const { if( i >= m_n ) throw std::out_of_range("string_view") ; return m_p[i] ; }
const Tchar * begin() const noexcept { return m_p ; } const Tchar * begin() const noexcept { return empty() ? nullptr : m_p ; }
const Tchar * cbegin() const noexcept { return m_p ; } const Tchar * cbegin() const noexcept { return empty() ? nullptr : m_p ; }
const Tchar * end() const noexcept { return m_p + m_n ; } const Tchar * end() const noexcept { return empty() ? nullptr : (m_p+m_n) ; }
const Tchar * cend() const noexcept { return m_p + m_n ; } const Tchar * cend() const noexcept { return empty() ? nullptr : (m_p+m_n) ; }
bool operator==( const basic_string_view<Tchar> & other ) const noexcept { return compare(other) == 0 ; } bool operator==( const basic_string_view<Tchar> & other ) const noexcept { return compare(other) == 0 ; }
bool operator!=( const basic_string_view<Tchar> & other ) const noexcept { return compare(other) != 0 ; } bool operator!=( const basic_string_view<Tchar> & other ) const noexcept { return compare(other) != 0 ; }
bool operator<( const basic_string_view<Tchar> & other ) const noexcept { return compare(other) < 0 ; } bool operator<( const basic_string_view<Tchar> & other ) const noexcept { return compare(other) < 0 ; }
@ -118,16 +118,17 @@ public:
bool operator>=( const basic_string_view<Tchar> & other ) const noexcept { return compare(other) >= 0 ; } bool operator>=( const basic_string_view<Tchar> & other ) const noexcept { return compare(other) >= 0 ; }
int compare( const basic_string_view<Tchar> & other ) const noexcept int compare( const basic_string_view<Tchar> & other ) const noexcept
{ {
int rc = std::char_traits<Tchar>::compare( m_p , other.m_p , std::min(m_n,other.m_n) ) ; int rc = ( empty() || other.empty() ) ? 0 : std::char_traits<Tchar>::compare( m_p , other.m_p , std::min(m_n,other.m_n) ) ;
return rc == 0 ? ( m_n < other.m_n ? -1 : (m_n==other.m_n?0:1) ) : rc ; return rc == 0 ? ( m_n < other.m_n ? -1 : (m_n==other.m_n?0:1) ) : rc ;
} }
string_view substr( std::size_t pos , std::size_t count = npos ) const string_view substr( std::size_t pos , std::size_t count = npos ) const
{ {
if( pos > m_n ) throw std::out_of_range( "string_view" ) ; if( empty() || pos > m_n ) throw std::out_of_range( "string_view" ) ;
return string_view( m_p + pos , std::min(m_n-pos,count) ) ; return string_view( m_p + pos , std::min(m_n-pos,count) ) ;
} }
std::size_t find( Tchar c ) const noexcept std::size_t find( Tchar c ) const noexcept
{ {
if( empty() ) return std::string::npos ;
const Tchar * p = m_p ; const Tchar * p = m_p ;
std::size_t n = m_n ; std::size_t n = m_n ;
for( std::size_t pos = 0U ; n ; p++ , n-- , pos++ ) for( std::size_t pos = 0U ; n ; p++ , n-- , pos++ )
@ -137,11 +138,28 @@ public:
} }
return std::string::npos ; return std::string::npos ;
} }
std::size_t find_first_of( basic_string_view<Tchar> chars ) const noexcept std::size_t find( const Tchar * substr_p , std::size_t pos , std::size_t substr_n ) const
{ {
const Tchar * p = m_p ; return find( basic_string_view<Tchar>(substr_p,substr_n) , pos ) ;
std::size_t n = m_n ; }
for( std::size_t pos = 0U ; n ; p++ , n-- , pos++ ) std::size_t find( basic_string_view<Tchar> substr , std::size_t pos = 0U ) const
{
if( empty() || pos >= m_n ) return std::string::npos ;
if( substr.empty() ) return pos ;
auto const end = m_p + m_n ;
auto p = std::search( m_p+pos , end , substr.m_p , substr.m_p+substr.m_n ) ;
return p == end ? std::string::npos : std::distance(m_p,p) ;
}
std::size_t find_first_of( const Tchar * chars , std::size_t pos , std::size_t chars_size ) const noexcept
{
return find_first_of( basic_string_view<Tchar>(chars,chars_size) , pos ) ;
}
std::size_t find_first_of( basic_string_view<Tchar> chars , std::size_t pos = 0U ) const noexcept
{
if( empty() || pos >= m_n || chars.empty() ) return std::string::npos ;
const Tchar * p = m_p + pos ;
std::size_t n = m_n - pos ;
for( ; n ; p++ , n-- , pos++ )
{ {
const std::size_t i_end = chars.size() ; const std::size_t i_end = chars.size() ;
for( std::size_t i = 0U ; i < i_end ; i++ ) for( std::size_t i = 0U ; i < i_end ; i++ )
@ -152,11 +170,16 @@ public:
} }
return std::string::npos ; return std::string::npos ;
} }
std::size_t find_first_not_of( basic_string_view<Tchar> chars ) const noexcept std::size_t find_first_not_of( const Tchar * chars , std::size_t pos , std::size_t chars_size ) const noexcept
{ {
const Tchar * p = m_p ; return find_first_not_of( basic_string_view<Tchar>(chars,chars_size) , pos ) ;
std::size_t n = m_n ; }
for( std::size_t pos = 0U ; n ; p++ , n-- , pos++ ) std::size_t find_first_not_of( basic_string_view<Tchar> chars , std::size_t pos = 0U ) const noexcept
{
if( empty() || pos >= m_n || chars.empty() ) return std::string::npos ;
const Tchar * p = m_p + pos ;
std::size_t n = m_n - pos ;
for( ; n ; p++ , n-- , pos++ )
{ {
bool match = false ; bool match = false ;
const std::size_t i_end = chars.size() ; const std::size_t i_end = chars.size() ;
@ -172,7 +195,7 @@ public:
} }
std::basic_string<Tchar> sv_to_string_imp() const std::basic_string<Tchar> sv_to_string_imp() const
{ {
return std::basic_string<Tchar>( m_p , m_n ) ; return empty() ? std::basic_string<Tchar>() : std::basic_string<Tchar>( m_p , m_n ) ;
} }
private: private:
@ -180,8 +203,6 @@ private:
std::size_t m_n{0U} ; std::size_t m_n{0U} ;
} ; } ;
static_assert( G::string_view("foo",nullptr).length() == 3U , "" ) ;
namespace G namespace G
{ {
template <typename Tchar> std::basic_string<Tchar> sv_to_string( basic_string_view<Tchar> sv ) template <typename Tchar> std::basic_string<Tchar> sv_to_string( basic_string_view<Tchar> sv )
@ -189,18 +210,38 @@ namespace G
// (greppable name -- remove when using c++17 std::string_view) // (greppable name -- remove when using c++17 std::string_view)
return std::basic_string<Tchar>( sv.sv_to_string_imp() ) ; return std::basic_string<Tchar>( sv.sv_to_string_imp() ) ;
} }
inline std::ostream & operator<<( std::ostream & stream , const string_view & s ) inline std::ostream & operator<<( std::ostream & stream , const string_view & sv )
{ {
return stream.write( s.data() , s.size() ) ; if( !sv.empty() )
stream.write( sv.data() , sv.size() ) ; // NOLINT narrowing
return stream ;
} }
inline std::wostream & operator<<( std::wostream & stream , const wstring_view & s ) inline std::wostream & operator<<( std::wostream & stream , const wstring_view & sv )
{ {
return stream.write( s.data() , s.size() ) ; if( !sv.empty() )
stream.write( sv.data() , sv.size() ) ; // NOLINT narrowing
return stream ;
} }
template <typename Tchar> void swap( basic_string_view<Tchar> & a , basic_string_view<Tchar> b ) noexcept template <typename Tchar> void swap( basic_string_view<Tchar> & a , basic_string_view<Tchar> b ) noexcept
{ {
a.swap( b ) ; a.swap( b ) ;
} }
inline bool operator==( const std::string & s , string_view sv )
{
return sv.empty() ? s.empty() : ( 0 == s.compare( 0 , s.size() , sv.data() , sv.size() ) ) ;
}
inline bool operator==( string_view sv , const std::string & s )
{
return sv.empty() ? s.empty() : ( 0 == s.compare( 0 , s.size() , sv.data() , sv.size() ) ) ;
}
inline bool operator!=( const std::string & s , string_view sv )
{
return !(s == sv) ;
}
inline bool operator!=( string_view sv , const std::string & s )
{
return !(sv == s) ;
}
} }
#endif #endif

View File

@ -25,11 +25,19 @@
#ifdef G_STR_IMP #ifdef G_STR_IMP
#undef G_STR_IMP #undef G_STR_IMP
#endif #endif
#define G_STR_IMP(X) #X
#ifdef G_STR #ifdef G_STR
#undef G_STR #undef G_STR
#endif #endif
#define G_STR(X) G_STR_IMP(X) #ifdef G_STR_PASTE_IMP
#undef G_STR_PASTE_IMP
#endif
#ifdef G_STR_PASTE
#undef G_STR_PASTE
#endif
#define G_STR_IMP(a) #a
#define G_STR(a) G_STR_IMP(a)
#define G_STR_PASTE_IMP(a,b) a##b
#define G_STR_PASTE(a,b) G_STR_PASTE_IMP(a,b)
#endif #endif

View File

@ -31,43 +31,31 @@ G::Time::Time( int hh , int mm , int ss ) :
{ {
} }
G::Time::Time( const BrokenDownTime & tm ) G::Time::Time( const BrokenDownTime & tm ) :
m_hh(tm.hour()) ,
m_mm(tm.min()) ,
m_ss(tm.sec())
{ {
m_hh = tm.hour() ;
m_mm = tm.min() ;
m_ss = tm.sec() ;
} }
G::Time::Time() G::Time::Time() :
Time(SystemTime::now().utc())
{ {
BrokenDownTime tm = SystemTime::now().utc() ;
m_hh = tm.hour() ;
m_mm = tm.min() ;
m_ss = tm.sec() ;
} }
G::Time::Time( SystemTime t ) G::Time::Time( SystemTime t ) :
Time(t.utc())
{ {
BrokenDownTime tm = t.utc() ;
m_hh = tm.hour() ;
m_mm = tm.min() ;
m_ss = tm.sec() ;
} }
G::Time::Time( const LocalTime & ) G::Time::Time( const LocalTime & ) :
Time(SystemTime::now().local())
{ {
BrokenDownTime tm = SystemTime::now().local() ;
m_hh = tm.hour() ;
m_mm = tm.min() ;
m_ss = tm.sec() ;
} }
G::Time::Time( SystemTime t , const LocalTime & ) G::Time::Time( SystemTime t , const LocalTime & ) :
Time(t.local())
{ {
BrokenDownTime tm = t.local() ;
m_hh = tm.hour() ;
m_mm = tm.min() ;
m_ss = tm.sec() ;
} }
int G::Time::hours() const int G::Time::hours() const
@ -121,9 +109,9 @@ G::Time G::Time::at( unsigned int s )
unsigned int hh = s / 3600U ; unsigned int hh = s / 3600U ;
unsigned int mm_ss = s - (hh*3600U) ; unsigned int mm_ss = s - (hh*3600U) ;
return { return {
std::max(0,std::min(static_cast<int>(hh),23)) , std::max(0,std::min(23,static_cast<int>(hh))) ,
std::max(0,std::min(static_cast<int>(mm_ss/60U),59)) , std::max(0,std::min(59,static_cast<int>(mm_ss/60U))) ,
std::max(0,std::min(static_cast<int>(mm_ss%60U),59)) } ; std::max(0,std::min(59,static_cast<int>(mm_ss%60U))) } ;
} }
bool G::Time::operator==( const Time & other ) const bool G::Time::operator==( const Time & other ) const

View File

@ -17,14 +17,6 @@
noinst_LIBRARIES = libgnet.a noinst_LIBRARIES = libgnet.a
if GCONFIG_IPV6
IP46_SOURCES = gaddress_ipv6.cpp
EXTRA_DIST_IP46 = gaddress_ipv4.cpp
else
IP46_SOURCES = gaddress_ipv4.cpp
EXTRA_DIST_IP46 = gaddress_ipv6.cpp
endif
if GCONFIG_WINDOWS if GCONFIG_WINDOWS
AM_CPPFLAGS = -I$(top_srcdir)/src/glib -I$(top_srcdir)/src/gssl -I$(top_srcdir)/src/win32 AM_CPPFLAGS = -I$(top_srcdir)/src/glib -I$(top_srcdir)/src/gssl -I$(top_srcdir)/src/win32
@ -41,7 +33,6 @@ if GCONFIG_INTERFACE_NAMES
EXTRA_DIST = \ EXTRA_DIST = \
$(EXTRA_DIST_COMMON) \ $(EXTRA_DIST_COMMON) \
$(EXTRA_DIST_IP46) \
ginterfaces_unix.cpp \ ginterfaces_unix.cpp \
ginterfaces_none.cpp ginterfaces_none.cpp
@ -53,7 +44,6 @@ else
EXTRA_DIST = \ EXTRA_DIST = \
$(EXTRA_DIST_COMMON) \ $(EXTRA_DIST_COMMON) \
$(EXTRA_DIST_IP46) \
ginterfaces_unix.cpp \ ginterfaces_unix.cpp \
ginterfaces_common.cpp \ ginterfaces_common.cpp \
ginterfaces_win32.cpp ginterfaces_win32.cpp
@ -64,13 +54,13 @@ INTERFACES_SOURCES = \
endif endif
libgnet_a_SOURCES = \ libgnet_a_SOURCES = \
$(IP46_SOURCES) \
gaddresslocal_none.cpp \ gaddresslocal_none.cpp \
gaddresslocal.h \ gaddresslocal.h \
gaddress4.cpp \ gaddress4.cpp \
gaddress4.h \ gaddress4.h \
gaddress6.cpp \ gaddress6.cpp \
gaddress6.h \ gaddress6.h \
gaddress.cpp \
gaddress.h \ gaddress.h \
gclient.cpp \ gclient.cpp \
gclient.h \ gclient.h \
@ -215,12 +205,11 @@ EXTRA_DIST = \
$(EXTRA_DIST_COMMON) \ $(EXTRA_DIST_COMMON) \
$(EXTRA_DIST_UDS) \ $(EXTRA_DIST_UDS) \
$(EXTRA_DIST_INTERFACES) \ $(EXTRA_DIST_INTERFACES) \
$(EXTRA_DIST_EVENTLOOP) \ $(EXTRA_DIST_EVENTLOOP)
$(EXTRA_DIST_IP46)
libgnet_a_SOURCES = \ libgnet_a_SOURCES = \
$(IP46_SOURCES) \
$(UDS_SOURCES) \ $(UDS_SOURCES) \
gaddress.cpp \
gaddress.h \ gaddress.h \
gaddresslocal.h \ gaddresslocal.h \
gaddress4.h \ gaddress4.h \

View File

@ -110,11 +110,11 @@ am__v_AR_0 = @echo " AR " $@;
am__v_AR_1 = am__v_AR_1 =
libgnet_a_AR = $(AR) $(ARFLAGS) libgnet_a_AR = $(AR) $(ARFLAGS)
libgnet_a_LIBADD = libgnet_a_LIBADD =
am__libgnet_a_SOURCES_DIST = gaddress_ipv4.cpp gaddress_ipv6.cpp \ am__libgnet_a_SOURCES_DIST = gaddresslocal_none.cpp \
gaddresslocal_none.cpp gaddresslocal_unix.cpp gaddress.h \ gaddresslocal_unix.cpp gaddress.cpp gaddress.h gaddresslocal.h \
gaddresslocal.h gaddress4.h gaddress4.cpp gaddress6.h \ gaddress4.h gaddress4.cpp gaddress6.h gaddress6.cpp \
gaddress6.cpp gclient.cpp gclient.h gclientptr.cpp \ gclient.cpp gclient.h gclientptr.cpp gclientptr.h \
gclientptr.h gconnection.cpp gconnection.h gdescriptor.h \ gconnection.cpp gconnection.h gdescriptor.h \
gdescriptor_unix.cpp gdnsblock.cpp gdnsblock.h gdnsmessage.cpp \ gdescriptor_unix.cpp gdnsblock.cpp gdnsblock.h gdnsmessage.cpp \
gdnsmessage.h gevent.h geventhandler.cpp geventhandler.h \ gdnsmessage.h gevent.h geventhandler.cpp geventhandler.h \
geventhandlerlist.cpp geventhandlerlist.h \ geventhandlerlist.cpp geventhandlerlist.h \
@ -135,20 +135,18 @@ am__libgnet_a_SOURCES_DIST = gaddress_ipv4.cpp gaddress_ipv6.cpp \
gtask.cpp gtask.h gtimer.cpp gtimer.h gtimerlist.cpp \ gtask.cpp gtask.h gtimer.cpp gtimer.h gtimerlist.cpp \
gtimerlist.h gdescriptor_win32.cpp geventloop_win32.cpp \ gtimerlist.h gdescriptor_win32.cpp geventloop_win32.cpp \
gfutureevent_win32.cpp gsocket_win32.cpp gfutureevent_win32.cpp gsocket_win32.cpp
@GCONFIG_IPV6_FALSE@am__objects_1 = gaddress_ipv4.$(OBJEXT) @GCONFIG_UDS_FALSE@@GCONFIG_WINDOWS_FALSE@am__objects_1 = gaddresslocal_none.$(OBJEXT)
@GCONFIG_IPV6_TRUE@am__objects_1 = gaddress_ipv6.$(OBJEXT) @GCONFIG_UDS_TRUE@@GCONFIG_WINDOWS_FALSE@am__objects_1 = gaddresslocal_unix.$(OBJEXT)
@GCONFIG_UDS_FALSE@@GCONFIG_WINDOWS_FALSE@am__objects_2 = gaddresslocal_none.$(OBJEXT) @GCONFIG_EPOLL_FALSE@@GCONFIG_WINDOWS_FALSE@am__objects_2 = geventloop_select.$(OBJEXT)
@GCONFIG_UDS_TRUE@@GCONFIG_WINDOWS_FALSE@am__objects_2 = gaddresslocal_unix.$(OBJEXT) @GCONFIG_EPOLL_TRUE@@GCONFIG_WINDOWS_FALSE@am__objects_2 = geventloop_epoll.$(OBJEXT)
@GCONFIG_EPOLL_FALSE@@GCONFIG_WINDOWS_FALSE@am__objects_3 = geventloop_select.$(OBJEXT) @GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_FALSE@am__objects_3 = ginterfaces_none.$(OBJEXT)
@GCONFIG_EPOLL_TRUE@@GCONFIG_WINDOWS_FALSE@am__objects_3 = geventloop_epoll.$(OBJEXT) @GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_TRUE@am__objects_3 = ginterfaces_none.$(OBJEXT)
@GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_FALSE@am__objects_4 = ginterfaces_none.$(OBJEXT) @GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_FALSE@am__objects_3 = ginterfaces_unix.$(OBJEXT) \
@GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_TRUE@am__objects_4 = ginterfaces_none.$(OBJEXT)
@GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_FALSE@am__objects_4 = ginterfaces_unix.$(OBJEXT) \
@GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_FALSE@ ginterfaces_common.$(OBJEXT) @GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_FALSE@ ginterfaces_common.$(OBJEXT)
@GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_TRUE@am__objects_4 = ginterfaces_common.$(OBJEXT) \ @GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_TRUE@am__objects_3 = ginterfaces_common.$(OBJEXT) \
@GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_TRUE@ ginterfaces_win32.$(OBJEXT) @GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_TRUE@ ginterfaces_win32.$(OBJEXT)
@GCONFIG_WINDOWS_FALSE@am_libgnet_a_OBJECTS = $(am__objects_1) \ @GCONFIG_WINDOWS_FALSE@am_libgnet_a_OBJECTS = $(am__objects_1) \
@GCONFIG_WINDOWS_FALSE@ $(am__objects_2) gaddress4.$(OBJEXT) \ @GCONFIG_WINDOWS_FALSE@ gaddress.$(OBJEXT) gaddress4.$(OBJEXT) \
@GCONFIG_WINDOWS_FALSE@ gaddress6.$(OBJEXT) gclient.$(OBJEXT) \ @GCONFIG_WINDOWS_FALSE@ gaddress6.$(OBJEXT) gclient.$(OBJEXT) \
@GCONFIG_WINDOWS_FALSE@ gclientptr.$(OBJEXT) \ @GCONFIG_WINDOWS_FALSE@ gclientptr.$(OBJEXT) \
@GCONFIG_WINDOWS_FALSE@ gconnection.$(OBJEXT) \ @GCONFIG_WINDOWS_FALSE@ gconnection.$(OBJEXT) \
@ -158,12 +156,12 @@ am__libgnet_a_SOURCES_DIST = gaddress_ipv4.cpp gaddress_ipv6.cpp \
@GCONFIG_WINDOWS_FALSE@ geventhandler.$(OBJEXT) \ @GCONFIG_WINDOWS_FALSE@ geventhandler.$(OBJEXT) \
@GCONFIG_WINDOWS_FALSE@ geventhandlerlist.$(OBJEXT) \ @GCONFIG_WINDOWS_FALSE@ geventhandlerlist.$(OBJEXT) \
@GCONFIG_WINDOWS_FALSE@ geventloggingcontext.$(OBJEXT) \ @GCONFIG_WINDOWS_FALSE@ geventloggingcontext.$(OBJEXT) \
@GCONFIG_WINDOWS_FALSE@ geventloop.$(OBJEXT) $(am__objects_3) \ @GCONFIG_WINDOWS_FALSE@ geventloop.$(OBJEXT) $(am__objects_2) \
@GCONFIG_WINDOWS_FALSE@ gexceptionhandler.$(OBJEXT) \ @GCONFIG_WINDOWS_FALSE@ gexceptionhandler.$(OBJEXT) \
@GCONFIG_WINDOWS_FALSE@ gexceptionsink.$(OBJEXT) \ @GCONFIG_WINDOWS_FALSE@ gexceptionsink.$(OBJEXT) \
@GCONFIG_WINDOWS_FALSE@ gexceptionsource.$(OBJEXT) \ @GCONFIG_WINDOWS_FALSE@ gexceptionsource.$(OBJEXT) \
@GCONFIG_WINDOWS_FALSE@ gfutureevent_unix.$(OBJEXT) \ @GCONFIG_WINDOWS_FALSE@ gfutureevent_unix.$(OBJEXT) \
@GCONFIG_WINDOWS_FALSE@ $(am__objects_4) glinebuffer.$(OBJEXT) \ @GCONFIG_WINDOWS_FALSE@ $(am__objects_3) glinebuffer.$(OBJEXT) \
@GCONFIG_WINDOWS_FALSE@ glinestore.$(OBJEXT) glocal.$(OBJEXT) \ @GCONFIG_WINDOWS_FALSE@ glinestore.$(OBJEXT) glocal.$(OBJEXT) \
@GCONFIG_WINDOWS_FALSE@ glocation.$(OBJEXT) gmonitor.$(OBJEXT) \ @GCONFIG_WINDOWS_FALSE@ glocation.$(OBJEXT) gmonitor.$(OBJEXT) \
@GCONFIG_WINDOWS_FALSE@ gmultiserver.$(OBJEXT) \ @GCONFIG_WINDOWS_FALSE@ gmultiserver.$(OBJEXT) \
@ -175,10 +173,11 @@ am__libgnet_a_SOURCES_DIST = gaddress_ipv4.cpp gaddress_ipv6.cpp \
@GCONFIG_WINDOWS_FALSE@ gsocketprotocol.$(OBJEXT) \ @GCONFIG_WINDOWS_FALSE@ gsocketprotocol.$(OBJEXT) \
@GCONFIG_WINDOWS_FALSE@ gsocks.$(OBJEXT) gtask.$(OBJEXT) \ @GCONFIG_WINDOWS_FALSE@ gsocks.$(OBJEXT) gtask.$(OBJEXT) \
@GCONFIG_WINDOWS_FALSE@ gtimer.$(OBJEXT) gtimerlist.$(OBJEXT) @GCONFIG_WINDOWS_FALSE@ gtimer.$(OBJEXT) gtimerlist.$(OBJEXT)
@GCONFIG_WINDOWS_TRUE@am_libgnet_a_OBJECTS = $(am__objects_1) \ @GCONFIG_WINDOWS_TRUE@am_libgnet_a_OBJECTS = \
@GCONFIG_WINDOWS_TRUE@ gaddresslocal_none.$(OBJEXT) \ @GCONFIG_WINDOWS_TRUE@ gaddresslocal_none.$(OBJEXT) \
@GCONFIG_WINDOWS_TRUE@ gaddress4.$(OBJEXT) gaddress6.$(OBJEXT) \ @GCONFIG_WINDOWS_TRUE@ gaddress4.$(OBJEXT) gaddress6.$(OBJEXT) \
@GCONFIG_WINDOWS_TRUE@ gclient.$(OBJEXT) gclientptr.$(OBJEXT) \ @GCONFIG_WINDOWS_TRUE@ gaddress.$(OBJEXT) gclient.$(OBJEXT) \
@GCONFIG_WINDOWS_TRUE@ gclientptr.$(OBJEXT) \
@GCONFIG_WINDOWS_TRUE@ gconnection.$(OBJEXT) \ @GCONFIG_WINDOWS_TRUE@ gconnection.$(OBJEXT) \
@GCONFIG_WINDOWS_TRUE@ gdescriptor_win32.$(OBJEXT) \ @GCONFIG_WINDOWS_TRUE@ gdescriptor_win32.$(OBJEXT) \
@GCONFIG_WINDOWS_TRUE@ gdnsblock.$(OBJEXT) \ @GCONFIG_WINDOWS_TRUE@ gdnsblock.$(OBJEXT) \
@ -193,7 +192,7 @@ am__libgnet_a_SOURCES_DIST = gaddress_ipv4.cpp gaddress_ipv6.cpp \
@GCONFIG_WINDOWS_TRUE@ gexceptionsink.$(OBJEXT) \ @GCONFIG_WINDOWS_TRUE@ gexceptionsink.$(OBJEXT) \
@GCONFIG_WINDOWS_TRUE@ gexceptionsource.$(OBJEXT) \ @GCONFIG_WINDOWS_TRUE@ gexceptionsource.$(OBJEXT) \
@GCONFIG_WINDOWS_TRUE@ gfutureevent_win32.$(OBJEXT) \ @GCONFIG_WINDOWS_TRUE@ gfutureevent_win32.$(OBJEXT) \
@GCONFIG_WINDOWS_TRUE@ $(am__objects_4) glinebuffer.$(OBJEXT) \ @GCONFIG_WINDOWS_TRUE@ $(am__objects_3) glinebuffer.$(OBJEXT) \
@GCONFIG_WINDOWS_TRUE@ glinestore.$(OBJEXT) glocal.$(OBJEXT) \ @GCONFIG_WINDOWS_TRUE@ glinestore.$(OBJEXT) glocal.$(OBJEXT) \
@GCONFIG_WINDOWS_TRUE@ glocation.$(OBJEXT) gmonitor.$(OBJEXT) \ @GCONFIG_WINDOWS_TRUE@ glocation.$(OBJEXT) gmonitor.$(OBJEXT) \
@GCONFIG_WINDOWS_TRUE@ gmultiserver.$(OBJEXT) \ @GCONFIG_WINDOWS_TRUE@ gmultiserver.$(OBJEXT) \
@ -221,9 +220,8 @@ am__v_at_1 =
DEFAULT_INCLUDES = -I.@am__isrc@ -I$(top_builddir)/src DEFAULT_INCLUDES = -I.@am__isrc@ -I$(top_builddir)/src
depcomp = $(SHELL) $(top_srcdir)/depcomp depcomp = $(SHELL) $(top_srcdir)/depcomp
am__maybe_remake_depfiles = depfiles am__maybe_remake_depfiles = depfiles
am__depfiles_remade = ./$(DEPDIR)/gaddress4.Po \ am__depfiles_remade = ./$(DEPDIR)/gaddress.Po ./$(DEPDIR)/gaddress4.Po \
./$(DEPDIR)/gaddress6.Po ./$(DEPDIR)/gaddress_ipv4.Po \ ./$(DEPDIR)/gaddress6.Po ./$(DEPDIR)/gaddresslocal_none.Po \
./$(DEPDIR)/gaddress_ipv6.Po ./$(DEPDIR)/gaddresslocal_none.Po \
./$(DEPDIR)/gaddresslocal_unix.Po ./$(DEPDIR)/gclient.Po \ ./$(DEPDIR)/gaddresslocal_unix.Po ./$(DEPDIR)/gclient.Po \
./$(DEPDIR)/gclientptr.Po ./$(DEPDIR)/gconnection.Po \ ./$(DEPDIR)/gclientptr.Po ./$(DEPDIR)/gconnection.Po \
./$(DEPDIR)/gdescriptor_unix.Po \ ./$(DEPDIR)/gdescriptor_unix.Po \
@ -432,10 +430,6 @@ top_build_prefix = @top_build_prefix@
top_builddir = @top_builddir@ top_builddir = @top_builddir@
top_srcdir = @top_srcdir@ top_srcdir = @top_srcdir@
noinst_LIBRARIES = libgnet.a noinst_LIBRARIES = libgnet.a
@GCONFIG_IPV6_FALSE@IP46_SOURCES = gaddress_ipv4.cpp
@GCONFIG_IPV6_TRUE@IP46_SOURCES = gaddress_ipv6.cpp
@GCONFIG_IPV6_FALSE@EXTRA_DIST_IP46 = gaddress_ipv6.cpp
@GCONFIG_IPV6_TRUE@EXTRA_DIST_IP46 = gaddress_ipv4.cpp
@GCONFIG_WINDOWS_FALSE@AM_CPPFLAGS = -I$(top_srcdir)/src/glib -I$(top_srcdir)/src/gssl @GCONFIG_WINDOWS_FALSE@AM_CPPFLAGS = -I$(top_srcdir)/src/glib -I$(top_srcdir)/src/gssl
@GCONFIG_WINDOWS_TRUE@AM_CPPFLAGS = -I$(top_srcdir)/src/glib -I$(top_srcdir)/src/gssl -I$(top_srcdir)/src/win32 @GCONFIG_WINDOWS_TRUE@AM_CPPFLAGS = -I$(top_srcdir)/src/glib -I$(top_srcdir)/src/gssl -I$(top_srcdir)/src/win32
@GCONFIG_WINDOWS_FALSE@EXTRA_DIST_COMMON = \ @GCONFIG_WINDOWS_FALSE@EXTRA_DIST_COMMON = \
@ -454,14 +448,12 @@ noinst_LIBRARIES = libgnet.a
@GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_TRUE@EXTRA_DIST = \ @GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_TRUE@EXTRA_DIST = \
@GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_TRUE@ $(EXTRA_DIST_COMMON) \ @GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_TRUE@ $(EXTRA_DIST_COMMON) \
@GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_TRUE@ $(EXTRA_DIST_IP46) \
@GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_TRUE@ ginterfaces_unix.cpp \ @GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_TRUE@ ginterfaces_unix.cpp \
@GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_TRUE@ ginterfaces_common.cpp \ @GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_TRUE@ ginterfaces_common.cpp \
@GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_TRUE@ ginterfaces_win32.cpp @GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_TRUE@ ginterfaces_win32.cpp
@GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_TRUE@EXTRA_DIST = \ @GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_TRUE@EXTRA_DIST = \
@GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_TRUE@ $(EXTRA_DIST_COMMON) \ @GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_TRUE@ $(EXTRA_DIST_COMMON) \
@GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_TRUE@ $(EXTRA_DIST_IP46) \
@GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_TRUE@ ginterfaces_unix.cpp \ @GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_TRUE@ ginterfaces_unix.cpp \
@GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_TRUE@ ginterfaces_none.cpp @GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_TRUE@ ginterfaces_none.cpp
@ -469,8 +461,7 @@ noinst_LIBRARIES = libgnet.a
@GCONFIG_WINDOWS_FALSE@ $(EXTRA_DIST_COMMON) \ @GCONFIG_WINDOWS_FALSE@ $(EXTRA_DIST_COMMON) \
@GCONFIG_WINDOWS_FALSE@ $(EXTRA_DIST_UDS) \ @GCONFIG_WINDOWS_FALSE@ $(EXTRA_DIST_UDS) \
@GCONFIG_WINDOWS_FALSE@ $(EXTRA_DIST_INTERFACES) \ @GCONFIG_WINDOWS_FALSE@ $(EXTRA_DIST_INTERFACES) \
@GCONFIG_WINDOWS_FALSE@ $(EXTRA_DIST_EVENTLOOP) \ @GCONFIG_WINDOWS_FALSE@ $(EXTRA_DIST_EVENTLOOP)
@GCONFIG_WINDOWS_FALSE@ $(EXTRA_DIST_IP46)
@GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_FALSE@INTERFACES_SOURCES = \ @GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_FALSE@INTERFACES_SOURCES = \
@GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_FALSE@ ginterfaces_none.cpp @GCONFIG_INTERFACE_NAMES_FALSE@@GCONFIG_WINDOWS_FALSE@ ginterfaces_none.cpp
@ -487,8 +478,8 @@ noinst_LIBRARIES = libgnet.a
@GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_TRUE@ ginterfaces_win32.cpp @GCONFIG_INTERFACE_NAMES_TRUE@@GCONFIG_WINDOWS_TRUE@ ginterfaces_win32.cpp
@GCONFIG_WINDOWS_FALSE@libgnet_a_SOURCES = \ @GCONFIG_WINDOWS_FALSE@libgnet_a_SOURCES = \
@GCONFIG_WINDOWS_FALSE@ $(IP46_SOURCES) \
@GCONFIG_WINDOWS_FALSE@ $(UDS_SOURCES) \ @GCONFIG_WINDOWS_FALSE@ $(UDS_SOURCES) \
@GCONFIG_WINDOWS_FALSE@ gaddress.cpp \
@GCONFIG_WINDOWS_FALSE@ gaddress.h \ @GCONFIG_WINDOWS_FALSE@ gaddress.h \
@GCONFIG_WINDOWS_FALSE@ gaddresslocal.h \ @GCONFIG_WINDOWS_FALSE@ gaddresslocal.h \
@GCONFIG_WINDOWS_FALSE@ gaddress4.h \ @GCONFIG_WINDOWS_FALSE@ gaddress4.h \
@ -565,13 +556,13 @@ noinst_LIBRARIES = libgnet.a
@GCONFIG_WINDOWS_FALSE@ gtimerlist.h @GCONFIG_WINDOWS_FALSE@ gtimerlist.h
@GCONFIG_WINDOWS_TRUE@libgnet_a_SOURCES = \ @GCONFIG_WINDOWS_TRUE@libgnet_a_SOURCES = \
@GCONFIG_WINDOWS_TRUE@ $(IP46_SOURCES) \
@GCONFIG_WINDOWS_TRUE@ gaddresslocal_none.cpp \ @GCONFIG_WINDOWS_TRUE@ gaddresslocal_none.cpp \
@GCONFIG_WINDOWS_TRUE@ gaddresslocal.h \ @GCONFIG_WINDOWS_TRUE@ gaddresslocal.h \
@GCONFIG_WINDOWS_TRUE@ gaddress4.cpp \ @GCONFIG_WINDOWS_TRUE@ gaddress4.cpp \
@GCONFIG_WINDOWS_TRUE@ gaddress4.h \ @GCONFIG_WINDOWS_TRUE@ gaddress4.h \
@GCONFIG_WINDOWS_TRUE@ gaddress6.cpp \ @GCONFIG_WINDOWS_TRUE@ gaddress6.cpp \
@GCONFIG_WINDOWS_TRUE@ gaddress6.h \ @GCONFIG_WINDOWS_TRUE@ gaddress6.h \
@GCONFIG_WINDOWS_TRUE@ gaddress.cpp \
@GCONFIG_WINDOWS_TRUE@ gaddress.h \ @GCONFIG_WINDOWS_TRUE@ gaddress.h \
@GCONFIG_WINDOWS_TRUE@ gclient.cpp \ @GCONFIG_WINDOWS_TRUE@ gclient.cpp \
@GCONFIG_WINDOWS_TRUE@ gclient.h \ @GCONFIG_WINDOWS_TRUE@ gclient.h \
@ -725,10 +716,9 @@ mostlyclean-compile:
distclean-compile: distclean-compile:
-rm -f *.tab.c -rm -f *.tab.c
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/gaddress.Po@am__quote@ # am--include-marker
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/gaddress4.Po@am__quote@ # am--include-marker @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/gaddress4.Po@am__quote@ # am--include-marker
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/gaddress6.Po@am__quote@ # am--include-marker @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/gaddress6.Po@am__quote@ # am--include-marker
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/gaddress_ipv4.Po@am__quote@ # am--include-marker
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/gaddress_ipv6.Po@am__quote@ # am--include-marker
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/gaddresslocal_none.Po@am__quote@ # am--include-marker @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/gaddresslocal_none.Po@am__quote@ # am--include-marker
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/gaddresslocal_unix.Po@am__quote@ # am--include-marker @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/gaddresslocal_unix.Po@am__quote@ # am--include-marker
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/gclient.Po@am__quote@ # am--include-marker @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/gclient.Po@am__quote@ # am--include-marker
@ -918,10 +908,9 @@ clean: clean-am
clean-am: clean-generic clean-noinstLIBRARIES mostlyclean-am clean-am: clean-generic clean-noinstLIBRARIES mostlyclean-am
distclean: distclean-am distclean: distclean-am
-rm -f ./$(DEPDIR)/gaddress4.Po -rm -f ./$(DEPDIR)/gaddress.Po
-rm -f ./$(DEPDIR)/gaddress4.Po
-rm -f ./$(DEPDIR)/gaddress6.Po -rm -f ./$(DEPDIR)/gaddress6.Po
-rm -f ./$(DEPDIR)/gaddress_ipv4.Po
-rm -f ./$(DEPDIR)/gaddress_ipv6.Po
-rm -f ./$(DEPDIR)/gaddresslocal_none.Po -rm -f ./$(DEPDIR)/gaddresslocal_none.Po
-rm -f ./$(DEPDIR)/gaddresslocal_unix.Po -rm -f ./$(DEPDIR)/gaddresslocal_unix.Po
-rm -f ./$(DEPDIR)/gclient.Po -rm -f ./$(DEPDIR)/gclient.Po
@ -1011,10 +1000,9 @@ install-ps-am:
installcheck-am: installcheck-am:
maintainer-clean: maintainer-clean-am maintainer-clean: maintainer-clean-am
-rm -f ./$(DEPDIR)/gaddress4.Po -rm -f ./$(DEPDIR)/gaddress.Po
-rm -f ./$(DEPDIR)/gaddress4.Po
-rm -f ./$(DEPDIR)/gaddress6.Po -rm -f ./$(DEPDIR)/gaddress6.Po
-rm -f ./$(DEPDIR)/gaddress_ipv4.Po
-rm -f ./$(DEPDIR)/gaddress_ipv6.Po
-rm -f ./$(DEPDIR)/gaddresslocal_none.Po -rm -f ./$(DEPDIR)/gaddresslocal_none.Po
-rm -f ./$(DEPDIR)/gaddresslocal_unix.Po -rm -f ./$(DEPDIR)/gaddresslocal_unix.Po
-rm -f ./$(DEPDIR)/gclient.Po -rm -f ./$(DEPDIR)/gclient.Po

View File

@ -15,7 +15,7 @@
// along with this program. If not, see <http://www.gnu.org/licenses/>. // along with this program. If not, see <http://www.gnu.org/licenses/>.
// === // ===
/// ///
/// \file gaddress_ipv6.cpp /// \file gaddress.cpp
/// ///
#include "gdef.h" #include "gdef.h"
@ -108,7 +108,8 @@ GNet::Address::Address( const sockaddr * addr , socklen_t len , bool ipv6_scope_
GNet::Address::Address( const std::string & s ) GNet::Address::Address( const std::string & s )
{ {
std::string r1 , r2 ; std::string r1 ;
std::string r2 ;
if( s.empty() ) if( s.empty() )
throw Address::Error( "empty string" ) ; throw Address::Error( "empty string" ) ;
else if( AddressLocal::af() && isFamilyLocal(s) ) else if( AddressLocal::af() && isFamilyLocal(s) )
@ -123,7 +124,8 @@ GNet::Address::Address( const std::string & s )
GNet::Address::Address( const std::string & s , NotLocal ) GNet::Address::Address( const std::string & s , NotLocal )
{ {
std::string r1 , r2 ; std::string r1 ;
std::string r2 ;
if( s.empty() ) if( s.empty() )
throw Address::Error( "empty string" ) ; throw Address::Error( "empty string" ) ;
else if( Address4::af() && Address4::validString(s,&r1) ) else if( Address4::af() && Address4::validString(s,&r1) )
@ -136,7 +138,8 @@ GNet::Address::Address( const std::string & s , NotLocal )
GNet::Address::Address( const std::string & host_part , const std::string & port_part ) GNet::Address::Address( const std::string & host_part , const std::string & port_part )
{ {
std::string r1 , r2 ; std::string r1 ;
std::string r2 ;
if( host_part.empty() ) if( host_part.empty() )
throw Address::Error( "empty string" ) ; throw Address::Error( "empty string" ) ;
else if( AddressLocal::af() && isFamilyLocal( host_part ) ) else if( AddressLocal::af() && isFamilyLocal( host_part ) )
@ -151,7 +154,8 @@ GNet::Address::Address( const std::string & host_part , const std::string & port
GNet::Address::Address( const std::string & host_part , unsigned int port ) GNet::Address::Address( const std::string & host_part , unsigned int port )
{ {
std::string r1 , r2 ; std::string r1 ;
std::string r2 ;
if( host_part.empty() ) if( host_part.empty() )
throw Address::Error( "empty string" ) ; throw Address::Error( "empty string" ) ;
else if( AddressLocal::af() && isFamilyLocal( host_part ) ) else if( AddressLocal::af() && isFamilyLocal( host_part ) )

View File

@ -1,421 +0,0 @@
//
// Copyright (C) 2001-2021 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 3 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, see <http://www.gnu.org/licenses/>.
// ===
///
/// \file gaddress_ipv4.cpp
///
#include "gdef.h"
#include "gaddress4.h"
#include "gaddress.h"
#include "gassert.h"
#include <algorithm> // std::swap()
#include <utility> // std::swap()
#include <cstring>
namespace GNet
{
class Address6
{
} ;
class AddressLocal
{
} ;
namespace AddressImp
{
void check( GNet::Address::Family f )
{
if( !Address::supports(f) )
throw Address::BadFamily() ;
}
}
}
bool GNet::Address::supports( Family f )
{
return f == Family::ipv4 ;
}
bool GNet::Address::supports( int af , int )
{
return af == AF_INET ;
}
bool GNet::Address::supports( const Address::Domain & , int domain )
{
return domain == Address4::domain() ;
}
GNet::Address GNet::Address::defaultAddress()
{
return { Family::ipv4 , 0U } ;
}
GNet::Address::Address( Family f , unsigned int port ) :
m_ipv4(std::make_unique<Address4>(port))
{
AddressImp::check( f ) ;
}
GNet::Address::Address( const AddressStorage & storage ) :
m_ipv4(std::make_unique<Address4>(storage.p(),storage.n()))
{
}
GNet::Address::Address( const sockaddr * addr , socklen_t len ) :
m_ipv4(std::make_unique<Address4>(addr,len))
{
}
GNet::Address::Address( const sockaddr * addr , socklen_t len , bool ) :
m_ipv4(std::make_unique<Address4>(addr,len))
{
}
GNet::Address::Address( const std::string & s ) :
m_ipv4(std::make_unique<Address4>(s))
{
}
GNet::Address::Address( const std::string & s , NotLocal ) :
m_ipv4(std::make_unique<Address4>(s))
{
}
GNet::Address::Address( const std::string & host_part , const std::string & port_part ) :
m_ipv4(std::make_unique<Address4>(host_part,port_part))
{
}
GNet::Address::Address( const std::string & s , unsigned int port ) :
m_ipv4(std::make_unique<Address4>(s,port))
{
}
GNet::Address::Address( Family f , unsigned int port , int loopback_overload ) :
m_ipv4(std::make_unique<Address4>(port,loopback_overload))
{
AddressImp::check( f ) ;
}
GNet::Address::Address( const Address & other ) :
m_ipv4(std::make_unique<Address4>(*other.m_ipv4))
{
}
GNet::Address::Address( Address && other ) noexcept
= default ;
GNet::Address::~Address()
= default;
void GNet::Address::swap( Address & other ) noexcept
{
using std::swap ;
swap( m_ipv4 , other.m_ipv4 ) ;
}
GNet::Address & GNet::Address::operator=( const Address & other )
{
Address(other).swap( *this ) ;
return *this ;
}
GNet::Address & GNet::Address::operator=( Address && other ) noexcept
= default ;
GNet::Address GNet::Address::parse( const std::string & s )
{
return Address( s ) ;
}
GNet::Address GNet::Address::parse( const std::string & s , Address::NotLocal not_local )
{
return { s , not_local } ;
}
GNet::Address GNet::Address::parse( const std::string & host_part , unsigned int port )
{
return { host_part , port } ;
}
GNet::Address GNet::Address::parse( const std::string & host_part , const std::string & port_part )
{
return { host_part , port_part } ;
}
bool GNet::Address::isFamilyLocal( const std::string & )
{
return false ;
}
GNet::Address GNet::Address::loopback( Family f , unsigned int port )
{
return Address( f , port , 1 ) ;
}
GNet::Address & GNet::Address::setPort( unsigned int port )
{
m_ipv4->setPort( port ) ;
return *this ;
}
GNet::Address & GNet::Address::setScopeId( unsigned long )
{
return *this ; // not relevant for ipv4
}
bool GNet::Address::setZone( const std::string & )
{
return true ; // not relevant for ipv4
}
unsigned int GNet::Address::bits() const
{
return m_ipv4->bits() ;
}
bool GNet::Address::isLoopback() const
{
return m_ipv4->isLoopback() ;
}
bool GNet::Address::isLocal( std::string & reason ) const
{
return m_ipv4->isLocal( reason ) ;
}
bool GNet::Address::isLinkLocal() const
{
return m_ipv4->isLinkLocal() ;
}
bool GNet::Address::isUniqueLocal() const
{
return m_ipv4->isUniqueLocal() ;
}
bool GNet::Address::isAny() const
{
return m_ipv4->isAny() ;
}
bool GNet::Address::is4() const
{
return true ;
}
bool GNet::Address::is6() const
{
return false ;
}
bool GNet::Address::same( const Address & other , bool ) const
{
return m_ipv4->same( *other.m_ipv4 ) ;
}
bool GNet::Address::operator==( const Address & other ) const
{
return m_ipv4->same( *other.m_ipv4 ) ;
}
bool GNet::Address::operator!=( const Address & other ) const
{
return !( *this == other ) ;
}
bool GNet::Address::sameHostPart( const Address & other ) const
{
return m_ipv4->sameHostPart(*other.m_ipv4) ;
}
std::string GNet::Address::displayString( bool ) const
{
return m_ipv4->displayString() ;
}
std::string GNet::Address::hostPartString( bool ) const
{
return m_ipv4->hostPartString() ;
}
std::string GNet::Address::queryString() const
{
return m_ipv4->queryString() ;
}
bool GNet::Address::validString( const std::string & s , std::string * reason_p )
{
return Address4::validString( s , reason_p ) ;
}
bool GNet::Address::validString( const std::string & s , NotLocal , std::string * reason_p )
{
return Address4::validString( s , reason_p ) ;
}
bool GNet::Address::validStrings( const std::string & s1 , const std::string & s2 , std::string * reason_p )
{
return Address4::validStrings( s1 , s2 , reason_p ) ;
}
sockaddr * GNet::Address::address()
{
return m_ipv4->address() ;
}
const sockaddr * GNet::Address::address() const
{
return m_ipv4->address() ;
}
socklen_t GNet::Address::length() const
{
return Address4::length() ;
}
unsigned int GNet::Address::port() const
{
return m_ipv4->port() ;
}
unsigned long GNet::Address::scopeId( unsigned long default_ ) const
{
return default_ ;
}
bool GNet::Address::validPort( unsigned int port )
{
return Address4::validPort( port ) ;
}
bool GNet::Address::validData( const sockaddr * addr , socklen_t len )
{
return Address4::validData( addr , len ) ;
}
int GNet::Address::domain( Family family )
{
return family == Family::ipv4 ? Address4::domain() : 0 ;
}
GNet::Address::Family GNet::Address::family() const
{
return Family::ipv4 ;
}
int GNet::Address::af() const
{
return AF_INET ;
}
G::StringArray GNet::Address::wildcards() const
{
return m_ipv4->wildcards() ;
}
// ==
//| \class GNet::AddressStorageImp
/// A pimple-pattern implementation class used by GNet::AddressStorage.
///
class GNet::AddressStorageImp
{
public:
sockaddr_storage u ;
socklen_t n ;
} ;
// ==
GNet::AddressStorage::AddressStorage() :
m_imp(std::make_unique<AddressStorageImp>())
{
static_assert( sizeof(Address4::sockaddr_type) <= sizeof(sockaddr_storage) , "" ) ;
static_assert( alignof(Address4::sockaddr_type) <= alignof(sockaddr_storage) , "" ) ;
m_imp->n = sizeof( sockaddr_storage ) ;
}
GNet::AddressStorage::~AddressStorage()
= default;
sockaddr * GNet::AddressStorage::p1()
{
return reinterpret_cast<sockaddr*>(&(m_imp->u)) ;
}
socklen_t * GNet::AddressStorage::p2()
{
return &m_imp->n ;
}
const sockaddr * GNet::AddressStorage::p() const
{
return reinterpret_cast<const sockaddr*>(&(m_imp->u)) ;
}
socklen_t GNet::AddressStorage::n() const
{
return m_imp->n ;
}
// ==
#if ! GCONFIG_HAVE_INET_PTON
// fallback implementation for inet_pton() -- see gdef.h
int GNet::inet_pton_imp( int f , const char * p , void * result )
{
if( p == nullptr || result == nullptr )
{
return 0 ; // just in case
}
else if( f == AF_INET )
{
sockaddr_in sa {} ;
sa.sin_family = AF_INET ;
sa.sin_addr.s_addr = inet_addr( p ) ;
std::memcpy( result , &sa.sin_addr , sizeof(sa.sin_addr) ) ;
return 1 ;
}
else
{
return -1 ;
}
}
#endif
#if ! GCONFIG_HAVE_INET_NTOP
// fallback implementation for inet_ntop() -- see gdef.h
const char * GNet::inet_ntop_imp( int f , void * ap , char * buffer , std::size_t n )
{
if( f == AF_INET )
{
std::ostringstream ss ;
struct in_addr a ;
std::memcpy( &a , ap , sizeof(a) ) ;
ss << inet_ntoa( a ) ; // ignore warnings - this is not used if inet_ntop is available
if( n <= ss.str().length() ) return nullptr ;
std::strncpy( buffer , ss.str().c_str() , n ) ;
return buffer ;
}
else
{
return nullptr ;
}
}
#endif

View File

@ -27,6 +27,7 @@
#include <vector> #include <vector>
#include <iomanip> #include <iomanip>
#include <sstream> #include <sstream>
#include <stdexcept>
GNet::DnsMessageRequest::DnsMessageRequest( const std::string & type , const std::string & hostname , unsigned int id ) GNet::DnsMessageRequest::DnsMessageRequest( const std::string & type , const std::string & hostname , unsigned int id )
{ {
@ -224,10 +225,14 @@ GNet::DnsMessage GNet::DnsMessage::rejection( const DnsMessage & message , unsig
void GNet::DnsMessage::reject( unsigned int rcode ) void GNet::DnsMessage::reject( unsigned int rcode )
{ {
m_buffer.at(2U) |= 0x80 ; // QR if( m_buffer.size() < 10U )
m_buffer.at(3U) &= 0xf0 ; m_buffer.at(3U) |= ( rcode & 0x0f ) ; // RCODE throw std::out_of_range( "dns message buffer too small" ) ;
m_buffer.at(6U) = 0U ; m_buffer.at(7U) = 0U ; // ANCOUNT
m_buffer.at(8U) = 0U ; m_buffer.at(9U) = 0U ; // NSCOUNT unsigned char * buffer = reinterpret_cast<unsigned char*>(&m_buffer[0]) ;
buffer[2U] |= 0x80U ; // QR
buffer[3U] &= 0xf0U ; buffer[3U] |= ( rcode & 0x0fU ) ; // RCODE
buffer[6U] = 0U ; buffer[7U] = 0U ; // ANCOUNT
buffer[8U] = 0U ; buffer[9U] = 0U ; // NSCOUNT
// chop off RRs // chop off RRs
unsigned int new_size = 12U ; // HEADER size unsigned int new_size = 12U ; // HEADER size
@ -266,7 +271,8 @@ GNet::Address GNet::DnsMessage::rrAddress( unsigned int record_index ) const
// == // ==
GNet::DnsMessageQuestion::DnsMessageQuestion( const DnsMessage & msg , unsigned int offset ) GNet::DnsMessageQuestion::DnsMessageQuestion( const DnsMessage & msg , unsigned int offset ) :
m_size(0U)
{ {
m_qname = DnsMessageNameParser::read( msg , offset ) ; m_qname = DnsMessageNameParser::read( msg , offset ) ;
m_size = DnsMessageNameParser::size( msg , offset ) + 2U + 2U ; // QNAME + QTYPE + QCLASS m_size = DnsMessageNameParser::size( msg , offset ) + 2U + 2U ; // QNAME + QTYPE + QCLASS
@ -341,6 +347,7 @@ GNet::DnsMessageRR::DnsMessageRR( const DnsMessage & msg , unsigned int offset )
m_offset(offset) , m_offset(offset) ,
m_size(0U) , m_size(0U) ,
m_type(0U) , m_type(0U) ,
m_class(0U) ,
m_rdata_offset(0U) , m_rdata_offset(0U) ,
m_rdata_size(0U) m_rdata_size(0U)
{ {

View File

@ -103,7 +103,7 @@ std::vector<GNet::Address> GNet::Interfaces::addresses( const G::StringArray & n
AddressList result ; AddressList result ;
for( const auto & name : names ) for( const auto & name : names )
{ {
if( Address::validStrings(name,G::Str::fromUInt(port)) ) if( Address::validStrings( name , G::Str::fromUInt(port) ) )
{ {
result.push_back( Address::parse(name,port) ) ; result.push_back( Address::parse(name,port) ) ;
} }

View File

@ -20,6 +20,7 @@
#include "gdef.h" #include "gdef.h"
#include "ginterfaces.h" #include "ginterfaces.h"
#include "gstr.h"
GNet::Interfaces::Interfaces() GNet::Interfaces::Interfaces()
= default; = default;

View File

@ -44,12 +44,7 @@ void GNet::LineBuffer::clear()
if( !transparent() ) if( !transparent() )
m_expect = 0U ; m_expect = 0U ;
G_ASSERT( m_in.empty() && empty() ) ; G_ASSERT( m_in.empty() && state().empty() ) ;
}
bool GNet::LineBuffer::empty() const
{
return state().empty() ;
} }
void GNet::LineBuffer::add( const char * data , std::size_t size ) void GNet::LineBuffer::add( const char * data , std::size_t size )
@ -244,8 +239,8 @@ GNet::LineBufferConfig::LineBufferConfig( const std::string & eol , std::size_t
GNet::LineBufferConfig GNet::LineBufferConfig::transparent() GNet::LineBufferConfig GNet::LineBufferConfig::transparent()
{ {
const std::size_t inf = ~(std::size_t(0)) ; static constexpr std::size_t inf = ~(std::size_t(0)) ;
//G_ASSERT( (inf+1U) == 0U ) ; static_assert( (inf+1U) == 0U , "" ) ;
return LineBufferConfig( std::string(1U,'\n') , 0U , 0U , inf ) ; return LineBufferConfig( std::string(1U,'\n') , 0U , 0U , inf ) ;
} }

View File

@ -188,9 +188,6 @@ public:
///< line ending. ///< line ending.
///< Precondition: more() ///< Precondition: more()
bool empty() const ;
///< Returns state().empty().
std::size_t eolsize() const ; std::size_t eolsize() const ;
///< Returns the size of line-ending associated with the ///< Returns the size of line-ending associated with the
///< current data(). This will be zero for a fixed-size ///< current data(). This will be zero for a fixed-size

View File

@ -336,7 +336,7 @@ std::size_t GNet::LineStore::find( const std::string & s , std::size_t startpos
std::size_t GNet::LineStore::search( std::string::const_iterator begin , std::string::const_iterator end , std::size_t GNet::LineStore::search( std::string::const_iterator begin , std::string::const_iterator end ,
std::size_t startpos ) const std::size_t startpos ) const
{ {
return std::search( LineStoreIterator(*this)+startpos , LineStoreIterator(*this,true) , begin , end ).pos() ; return std::search( LineStoreIterator(*this)+startpos , LineStoreIterator(*this,true) , begin , end ).pos() ; // NOLINT narrowing
} }
std::size_t GNet::LineStore::findSubStringAtEnd( const std::string & s , std::size_t startpos ) const std::size_t GNet::LineStore::findSubStringAtEnd( const std::string & s , std::size_t startpos ) const
@ -359,7 +359,7 @@ std::size_t GNet::LineStore::findSubStringAtEnd( const std::string & s , std::si
{ {
// compare leading substring with the end of the store // compare leading substring with the end of the store
const LineStoreIterator end( *this , true ) ; const LineStoreIterator end( *this , true ) ;
LineStoreIterator p = end - s_size ; LineStoreIterator p = end - s_size ; // NOLINT narrowing
if( imp::std_equal(s_start,s_end,p,end) ) if( imp::std_equal(s_start,s_end,p,end) )
{ {
result = p.pos() ; result = p.pos() ;

View File

@ -22,7 +22,6 @@
#include "glocal.h" #include "glocal.h"
#include "ghostname.h" #include "ghostname.h"
#include "gresolver.h" #include "gresolver.h"
#include "ginterfaces.h"
#include "glog.h" #include "glog.h"
#include <sstream> #include <sstream>

View File

@ -44,8 +44,8 @@ GNet::MultiServer::MultiServer( ExceptionSink es , const G::StringArray & interf
m_interfaces.erase( std::unique(m_interfaces.begin(),m_interfaces.end()) , m_interfaces.end() ) ; m_interfaces.erase( std::unique(m_interfaces.begin(),m_interfaces.end()) , m_interfaces.end() ) ;
// build a listening address list from explicit addresses and/or interface names // build a listening address list from explicit addresses and/or interface names
G::StringArray empty_names ; // interface names having no addresses
G::StringArray used_names ; // interface names having one or more addresses G::StringArray used_names ; // interface names having one or more addresses
G::StringArray empty_names ; // interface names having no addresses
G::StringArray bad_names ; // non-address non-interface names G::StringArray bad_names ; // non-address non-interface names
AddressList address_list = addresses( port , used_names , empty_names , bad_names ) ; AddressList address_list = addresses( port , used_names , empty_names , bad_names ) ;

View File

@ -29,7 +29,7 @@
GNet::ResolverFuture::ResolverFuture( const std::string & host , const std::string & service , int family , GNet::ResolverFuture::ResolverFuture( const std::string & host , const std::string & service , int family ,
bool dgram , bool for_async_hint ) : bool dgram , bool for_async_hint ) :
m_numeric_service(false) , m_numeric_service(!service.empty() && G::Str::isNumeric(service)) ,
m_socktype(dgram?SOCK_DGRAM:SOCK_STREAM) , m_socktype(dgram?SOCK_DGRAM:SOCK_STREAM) ,
m_host(host) , m_host(host) ,
m_host_p(m_host.c_str()) , m_host_p(m_host.c_str()) ,
@ -40,9 +40,8 @@ GNet::ResolverFuture::ResolverFuture( const std::string & host , const std::stri
m_rc(0) , m_rc(0) ,
m_ai(nullptr) m_ai(nullptr)
{ {
m_numeric_service = !service.empty() && G::Str::isNumeric(service) ;
std::memset( &m_ai_hint , 0 , sizeof(m_ai_hint) ) ; std::memset( &m_ai_hint , 0 , sizeof(m_ai_hint) ) ;
m_ai_hint.ai_flags = AI_CANONNAME | m_ai_hint.ai_flags = AI_CANONNAME | // NOLINT
( family == AF_UNSPEC ? AI_ADDRCONFIG : 0 ) | ( family == AF_UNSPEC ? AI_ADDRCONFIG : 0 ) |
( m_numeric_service ? AI_NUMERICSERV : 0 ) ; ( m_numeric_service ? AI_NUMERICSERV : 0 ) ;
m_ai_hint.ai_family = family ; m_ai_hint.ai_family = family ;

View File

@ -31,7 +31,7 @@ namespace GNet
{ {
namespace StreamSocketImp /// An implementation namespace for G::StreamSocket. namespace StreamSocketImp /// An implementation namespace for G::StreamSocket.
{ {
struct Options /// StreamSocket options struct Options /// StreamSocket options.
{ {
enum class Linger { default_ , zero , nolinger } ; enum class Linger { default_ , zero , nolinger } ;
Linger create_linger {Linger::nolinger} ; Linger create_linger {Linger::nolinger} ;
@ -42,7 +42,7 @@ namespace GNet
} }
namespace SocketImp /// An implementation namespace for G::Socket. namespace SocketImp /// An implementation namespace for G::Socket.
{ {
struct Options /// Socket options struct Options /// Socket options.
{ {
bool connect_pureipv6 {true} ; bool connect_pureipv6 {true} ;
bool bind_pureipv6 {true} ; bool bind_pureipv6 {true} ;

View File

@ -32,6 +32,7 @@
#include "glog.h" #include "glog.h"
#include <memory> #include <memory>
#include <numeric> #include <numeric>
#include <algorithm>
//| \class GNet::SocketProtocolImp //| \class GNet::SocketProtocolImp
/// A pimple-pattern implementation class used by GNet::SocketProtocol. /// A pimple-pattern implementation class used by GNet::SocketProtocol.
@ -165,9 +166,7 @@ GNet::SocketProtocolImp::~SocketProtocolImp()
void GNet::SocketProtocolImp::setReadBufferSize( std::size_t n ) void GNet::SocketProtocolImp::setReadBufferSize( std::size_t n )
{ {
m_read_buffer_size = n ; m_read_buffer_size = std::max( std::size_t(1U) , n ) ;
if( m_read_buffer_size == 0U )
m_read_buffer_size = 1U ;
} }
void GNet::SocketProtocolImp::onSecureConnectionTimeout() void GNet::SocketProtocolImp::onSecureConnectionTimeout()

View File

@ -30,7 +30,7 @@ GNet::Socks::Socks( const Location & location ) :
if( location.socks() ) if( location.socks() )
{ {
unsigned int far_port = location.socksFarPort() ; unsigned int far_port = location.socksFarPort() ;
if( !Address::validPort(far_port) ) if( !Address::validPort(far_port) || far_port > 0xffffU )
throw SocksError( "invalid port" ) ; throw SocksError( "invalid port" ) ;
m_request = buildPdu( location.socksFarHost() , far_port ) ; m_request = buildPdu( location.socksFarHost() , far_port ) ;

View File

@ -138,11 +138,10 @@ const GNet::TimerBase * GNet::TimerList::findSoonest() const
{ {
G_ASSERT( !m_locked ) ; G_ASSERT( !m_locked ) ;
TimerBase * result = nullptr ; TimerBase * result = nullptr ;
auto end = m_list.cend() ; for( const auto & t : m_list )
for( auto p = m_list.cbegin() ; p != end ; ++p )
{ {
if( (*p).m_timer != nullptr && (*p).m_timer->active() && ( result == nullptr || (*p).m_timer->t() < result->t() ) ) if( t.m_timer != nullptr && t.m_timer->active() && ( result == nullptr || t.m_timer->t() < result->t() ) )
result = (*p).m_timer ; result = t.m_timer ;
} }
return result ; return result ;
} }

View File

@ -179,7 +179,7 @@ bool GPop::ServerProtocol::sendContentLine( std::string & line , bool & stop )
// read the line of text // read the line of text
line.erase( 1U ) ; // leave "." line.erase( 1U ) ; // leave "."
G::Str::readLineFrom( *(m_content.get()) , crlf() , line , false/*erase*/ ) ; G::Str::readLineFrom( *m_content , crlf() , line , false/*erase*/ ) ;
// add crlf and choose an offset // add crlf and choose an offset
bool eof = m_content->fail() || m_content->bad() ; bool eof = m_content->fail() || m_content->bad() ;

View File

@ -222,7 +222,7 @@ void GSmtp::AdminServerPeer::flush()
else else
{ {
m_client_ptr.reset( std::make_unique<GSmtp::Client>( GNet::ExceptionSink(m_client_ptr,m_es.esrc()) , m_client_ptr.reset( std::make_unique<GSmtp::Client>( GNet::ExceptionSink(m_client_ptr,m_es.esrc()) ,
GNet::Location(m_remote_address) , m_server.clientSecrets() , m_server.clientConfig() ) ) ; m_server.ff() , GNet::Location(m_remote_address) , m_server.clientSecrets() , m_server.clientConfig() ) ) ;
m_client_ptr->sendMessagesFrom( m_server.store() ) ; // once connected m_client_ptr->sendMessagesFrom( m_server.store() ) ; // once connected
// no sendLine() -- sends "OK" or "error:" when complete -- see AdminServerPeer::clientDone() // no sendLine() -- sends "OK" or "error:" when complete -- see AdminServerPeer::clientDone()
@ -317,7 +317,7 @@ void GSmtp::AdminServerPeer::sendList( std::shared_ptr<MessageStore::Iterator> i
std::unique_ptr<StoredMessage> message( ++iter ) ; std::unique_ptr<StoredMessage> message( ++iter ) ;
if( message == nullptr ) break ; if( message == nullptr ) break ;
if( !first ) ss << eol() ; if( !first ) ss << eol() ;
ss << message->name() ; ss << message->id().str() ;
} }
std::string result = ss.str() ; std::string result = ss.str() ;
@ -340,9 +340,9 @@ bool GSmtp::AdminServerPeer::notifying() const
// === // ===
GSmtp::AdminServer::AdminServer( GNet::ExceptionSink es , MessageStore & store , GSmtp::AdminServer::AdminServer( GNet::ExceptionSink es , MessageStore & store ,
G::Slot::Signal<const std::string&> & forward_request , FilterFactory & ff , G::Slot::Signal<const std::string&> & forward_request ,
const GNet::ServerPeerConfig & server_peer_config , const GNet::ServerConfig & server_config , const GNet::ServerPeerConfig & server_peer_config , const GNet::ServerConfig & server_config ,
const GSmtp::Client::Config & client_config , const GAuth::Secrets & client_secrets , const GSmtp::Client::Config & client_config , const GAuth::SaslClientSecrets & client_secrets ,
const G::StringArray & interfaces , unsigned int port , bool allow_remote , const G::StringArray & interfaces , unsigned int port , bool allow_remote ,
const std::string & remote_address , unsigned int connection_timeout , const std::string & remote_address , unsigned int connection_timeout ,
const G::StringMap & info_commands , const G::StringMap & config_commands , const G::StringMap & info_commands , const G::StringMap & config_commands ,
@ -350,6 +350,7 @@ GSmtp::AdminServer::AdminServer( GNet::ExceptionSink es , MessageStore & store ,
GNet::MultiServer(es,interfaces,port,"admin",server_peer_config,server_config) , GNet::MultiServer(es,interfaces,port,"admin",server_peer_config,server_config) ,
m_forward_timer(*this,&AdminServer::onForwardTimeout,es) , m_forward_timer(*this,&AdminServer::onForwardTimeout,es) ,
m_store(store) , m_store(store) ,
m_ff(ff) ,
m_forward_request(forward_request) , m_forward_request(forward_request) ,
m_client_config(client_config) , m_client_config(client_config) ,
m_client_secrets(client_secrets) , m_client_secrets(client_secrets) ,
@ -436,7 +437,12 @@ GSmtp::MessageStore & GSmtp::AdminServer::store()
return m_store ; return m_store ;
} }
const GAuth::Secrets & GSmtp::AdminServer::clientSecrets() const GSmtp::FilterFactory & GSmtp::AdminServer::ff()
{
return m_ff ;
}
const GAuth::SaslClientSecrets & GSmtp::AdminServer::clientSecrets() const
{ {
return m_client_secrets ; return m_client_secrets ;
} }

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