emailrelay/src/main/mailrelay.cpp
Graeme Walker 6b2298628a v0.9.3
2001-10-21 12:00:00 +00:00

460 lines
11 KiB
C++

//
// Copyright (C) 2001 Graeme Walker <graeme_walker@users.sourceforge.net>
//
// This program is free software; you can redistribute it and/or
// modify it under the terms of the GNU General Public License
// as published by the Free Software Foundation; either
// version 2 of the License, or (at your option) any later
// version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
//
// ===
//
// mailrelay.cpp
//
#include "gdef.h"
#include "gsmtp.h"
#include "gsmtpserver.h"
#include "gsmtpclient.h"
#include "gevent.h"
#include "garg.h"
#include "gdaemon.h"
#include "gstr.h"
#include "gpath.h"
#include "gfilestore.h"
#include "gnewfile.h"
#include "gadminserver.h"
#include "gexception.h"
#include "gprocess.h"
#include "gmemory.h"
#include "ggetopt.h"
#include "gdebug.h"
#include <iostream>
#include <exception>
namespace
{
class Main
{
public:
explicit Main( G::Arg & arg ) ;
void run() ;
private:
std::string versionNumber() const ;
void version() const ;
void banner() const ;
void warranty() const ;
void usage( std::ostream & s , const std::string & exe , const G::GetOpt & opt ) const ;
void help( const std::string & exe ) const ;
void shortHelp( std::ostream & stream , const std::string & exe ) const ;
void copyright() const ;
std::string switchSpec() const ;
void runCore() ;
unsigned int ttyColumns() const ;
std::string checkOptions() const ;
const G::GetOpt & opt() const ;
unsigned int optPort() const ;
unsigned int optAdminPort() const ;
bool optCloseStderr() const ;
bool optImmediate() const ;
bool optLog() const ;
bool optSyslog() const ;
bool optDaemon() const ;
bool optDoForwarding() const ;
bool optDoServing() const ;
bool optDoAdmin() const ;
bool optAllowRemoteClients() const ;
G::Path optSpoolDir() const ;
std::string optServerAddress() const ;
void doForwarding( GSmtp::MessageStore & store ) ;
void closeFiles() ;
void closeMoreFiles() ;
std::string smtpIdent() const ;
void recordPid() ;
private:
G::Arg m_arg ;
G::GetOpt * m_opt ;
} ;
} ;
Main::Main( G::Arg & arg ) :
m_arg(arg)
{
}
void Main::banner() const
{
std::cout
<< "E-MailRelay V" << versionNumber() << std::endl ;
}
void Main::copyright() const
{
std::cout << "Copyright (C) 2001 Graeme Walker" << std::endl ;
}
void Main::warranty() const
{
std::cout
<< "This software is provided without warranty of any kind." << std::endl
<< "You may redistribure copies of this program under " << std::endl
<< "the terms of the GNU General Public License." << std::endl
<< "For more information refer to the file named COPYING." << std::endl ;
}
void Main::version() const
{
banner() ;
warranty() ;
copyright() ;
}
std::string Main::versionNumber() const
{
return "0.9.3" ;
}
std::string Main::smtpIdent() const
{
return std::string("E-MailRelay V") + versionNumber() ;
}
unsigned int Main::ttyColumns() const
{
const unsigned int default_ = 79U ;
try
{
const char * p = std::getenv( "COLUMNS" ) ;
if( p == NULL )
return default_ ;
else
return G::Str::toUInt(p) ;
}
catch( std::exception & )
{
return default_ ;
}
}
void Main::usage( std::ostream & s , const std::string & exe , const G::GetOpt & opt ) const
{
opt.showUsage( s , exe , "" , 30U , ttyColumns() ) ;
}
void Main::help( const std::string & exe ) const
{
std::cout << std::endl ;
std::cout
<< "To start a 'storage' daemon in background..." << std::endl
<< " " << exe << " --as-server" << std::endl
<< std::endl ;
std::cout
<< "To forward stored mail to \"mail.myisp.co.uk\"..." << std::endl
<< " " << exe << " --as-client mail.myisp.co.uk:smtp" << std::endl
<< std::endl ;
std::cout
<< "To run as a proxy (on port 10025) to a local server (on port 25)..." << std::endl
<< " " << exe << " --port 10025 --as-proxy localhost:25" << std::endl
<< std::endl ;
}
void Main::shortHelp( std::ostream & stream , const std::string & exe ) const
{
stream
<< std::string(exe.length()+2U,' ')
<< "try \"" << exe << " --help\" for more information" << std::endl ;
}
std::string Main::switchSpec() const
{
std::stringstream ss ;
ss
<< "h!help!displays help text and exits!0!|"
<< "l!log!writes log information on standard error (if open) and syslog (if not disabled)!0!|"
<< "v!verbose!generates more verbose logging "
<< "(if compiled-in and logging enabled and stderr open)!0!|"
<< "e!close-stderr!closes the standard error stream when daemonising!0!|"
<< "s!spool-dir!specifies the spool directory "
<< "(default is \"" << GSmtp::MessageStore::defaultDirectory()
<< "\")!1!dir|"
<< "q!as-client!equivalent to \"--no-syslog --no-daemon --log --dont-serve --forward --forward-to\"!"
<< "1!host:port|"
<< "d!as-server!equivalent to \"--close-stderr --log\"!0!|"
<< "m!immediate!forwards each message as soon as it is received (requires --forward-to)!0!|"
<< "n!no-syslog!disables syslog output!0!|"
<< "t!no-daemon!does not detach from the terminal!0!|"
<< "x!dont-serve!stops the process acting as a server (usually used with --forward)!0!|"
<< "f!forward!forwards stored mail on startup (requires --forward-to)!0!|"
<< "o!forward-to!specifies the remote smtp server (required by --forward and --admin)!1!host:port|"
<< "r!remote-clients!allows remote clients to connect!0!|"
<< "i!pid-file!records the daemon process-id in the given file!1!pid-file|"
<< "p!port!specifies the smtp listening port number!1!port|"
<< "a!admin!enables the administration interface and specifies its listening port number!1!admin-port|"
<< "y!as-proxy!equivalent to \"--close-stderr --log --immediate --forward-to\"!1!host:port|"
<< "z!filter!defines a mail pre-processor (disallowed if running as root)!1!program|"
<< "V!version!displays version information and exits!0!"
;
return ss.str() ;
}
const G::GetOpt & Main::opt() const
{
G_ASSERT( m_opt != NULL ) ;
return *m_opt ;
}
void Main::run()
{
m_opt = new G::GetOpt( m_arg , switchSpec() , '|' , '!' , '^' ) ;
if( opt().contains("help") )
{
banner() ;
std::cout << std::endl ;
usage( std::cout , m_arg.prefix() , opt() ) ;
help( m_arg.prefix() ) ;
copyright() ;
}
else if( opt().hasErrors() )
{
opt().showErrors( std::cerr , m_arg.prefix() ) ;
shortHelp( std::cerr , m_arg.prefix() ) ;
}
else if( opt().args().c() > 1U )
{
std::cerr << m_arg.prefix() << ": usage error: too many non-switch arguments" << std::endl ;
shortHelp( std::cerr , m_arg.prefix() ) ;
}
else if( opt().contains("version") )
{
version() ;
}
else
{
G::LogOutput debug( optLog() , opt().contains("verbose") ) ;
if( optSyslog() )
debug.syslog(G::LogOutput::Mail) ;
runCore() ;
}
}
bool Main::optLog() const
{
return
opt().contains("log") ||
opt().contains("as-client") ||
opt().contains("as-proxy") ||
opt().contains("as-server") ;
}
bool Main::optSyslog() const
{
return !opt().contains("no-syslog") && !opt().contains("as-client") ;
}
unsigned int Main::optPort() const
{
return opt().contains("port") ?
G::Str::toUInt(opt().value("port")) : 25U ;
}
unsigned int Main::optAdminPort() const
{
return opt().contains("admin") ?
G::Str::toUInt(opt().value("admin")) : 10025U ;
}
bool Main::optCloseStderr() const
{
return
opt().contains("close-stderr") ||
opt().contains("as-proxy") ||
opt().contains("as-server") ;
}
bool Main::optImmediate() const
{
return
opt().contains("immediate") ||
opt().contains("as-proxy") ;
}
bool Main::optDaemon() const
{
return !opt().contains("no-daemon") && !opt().contains("as-client") ;
}
G::Path Main::optSpoolDir() const
{
return opt().contains("spool-dir") ?
G::Path(opt().value("spool-dir")) :
GSmtp::MessageStore::defaultDirectory() ;
}
std::string Main::optServerAddress() const
{
const char * key = "forward-to" ;
if( opt().contains("as-client") )
key = "as-client" ;
else if( opt().contains("as-proxy") )
key = "as-proxy" ;
return opt().contains(key) ? opt().value(key) : std::string() ;
}
std::string Main::checkOptions() const
{
if( optDoAdmin() && optAdminPort() == optPort() )
{
return "the smtp listening port and the "
"admin listening port must be different" ;
}
if( optDaemon() && optSpoolDir().isRelative() )
{
return "in daemon mode the spool-dir must "
"be an absolute path (starting with /)" ;
}
if( !opt().contains("forward-to") && (
opt().contains("forward") ||
opt().contains("immediate") ||
opt().contains("admin") ) )
{
return "usage error: the --forward, --immediate and --admin "
"switches require --forward-to" ;
}
return std::string() ;
}
bool Main::optDoForwarding() const
{
return opt().contains("forward") || opt().contains("as-client") ;
}
void Main::doForwarding( GSmtp::MessageStore & store )
{
const bool quit_on_disconnect = true ;
GSmtp::Client client( store , quit_on_disconnect ) ;
std::string error = client.init( optServerAddress() ) ;
if( error.length() )
throw G::Exception( error + ": " + optServerAddress() ) ;
GNet::EventSources::instance().run() ;
}
void Main::closeFiles()
{
if( optDaemon() )
{
const bool keep_stderr = true ;
G::Process::closeFiles( keep_stderr ) ;
}
}
void Main::closeMoreFiles()
{
if( optDaemon() && optCloseStderr() )
G::Process::closeStderr() ;
}
bool Main::optDoServing() const
{
return !opt().contains("dont-serve") && !opt().contains("as-client") ;
}
bool Main::optAllowRemoteClients() const
{
return opt().contains("remote-clients") ;
}
bool Main::optDoAdmin() const
{
return opt().contains("admin") ;
}
void Main::runCore()
{
std::string error = checkOptions() ;
if( !error.empty() )
throw G::Exception( error ) ;
G::Daemon::PidFile pid_file ;
G::Process::setUmask() ;
if( optDaemon() )
{
closeFiles() ; // before opening any sockets or message-store streams
if( opt().contains("pid-file") )
pid_file = G::Daemon::PidFile( G::Path(opt().value("pid-file")) ) ;
G::Daemon::detach( pid_file ) ;
}
GSmtp::FileStore store( optSpoolDir() ) ;
if( opt().contains("filter") )
GSmtp::NewFile::setPreprocessor( G::Path(opt().value("filter")) ) ;
std::auto_ptr<GNet::EventSources> event_loop( GNet::EventSources::create() ) ;
if( ! event_loop->init() )
throw G::Exception( "cannot initialise network layer" ) ;
if( optDoForwarding() )
{
if( store.empty() )
std::cerr << m_arg.prefix() << ": no messages to send" << std::endl ;
else
doForwarding( store ) ;
}
if( optDoServing() )
{
GSmtp::Server server( optPort() , optAllowRemoteClients() , smtpIdent() ,
optImmediate() ? optServerAddress() : std::string() ) ;
std::auto_ptr<GSmtp::AdminServer> admin_server ;
if( optDoAdmin() )
{
admin_server <<= new GSmtp::AdminServer( optAdminPort() ,
optAllowRemoteClients() , optServerAddress() ) ;
}
pid_file.commit() ;
closeMoreFiles() ;
event_loop->run() ;
}
}
int main( int argc , char * argv [] )
{
G::Arg arg( argc , argv ) ;
try
{
Main main( arg ) ;
main.run() ;
return EXIT_SUCCESS ;
}
catch( std::exception & e )
{
std::cerr << arg.prefix() << ": exception: " << e.what() << std::endl ;
}
catch( ... )
{
std::cerr << arg.prefix() << ": unrecognised exception" << std::endl ;
}
return EXIT_FAILURE ;
}