emailrelay/src/main/run.cpp
Graeme Walker 61ffec9a36 v1.8.1
2008-05-21 12:00:00 +00:00

651 lines
16 KiB
C++

//
// Copyright (C) 2001-2008 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/>.
// ===
//
// run.cpp
//
#include "gdef.h"
#include "gssl.h"
#include "gsmtp.h"
#include "run.h"
#include "admin.h"
#include "gsmtpserver.h"
#include "gsmtpclient.h"
#include "gsasl.h"
#include "gsecrets.h"
#include "geventloop.h"
#include "garg.h"
#include "gdaemon.h"
#include "gpidfile.h"
#include "gfilestore.h"
#include "gnewfile.h"
#include "gadminserver.h"
#include "gpopserver.h"
#include "gprocessorfactory.h"
#include "gverifierfactory.h"
#include "gslot.h"
#include "gmonitor.h"
#include "glocal.h"
#include "gfile.h"
#include "gpath.h"
#include "groot.h"
#include "gexception.h"
#include "gprocess.h"
#include "gmemory.h"
#include "gtest.h"
#include "glogoutput.h"
#include "gdebug.h"
#include "legal.h"
#include <iostream>
#include <exception>
#include <utility>
std::string Main::Run::versionNumber()
{
return "1.8.1" ;
}
Main::Run::Run( Main::Output & output , const G::Arg & arg , const std::string & switch_spec ) :
m_output(output) ,
m_switch_spec(switch_spec) ,
m_arg(arg) ,
m_client_resolver_info(std::string(),std::string()) ,
m_prepare_error(false)
{
m_client.doneSignal().connect( G::slot(*this,&Run::pollingClientDone) ) ;
m_client.eventSignal().connect( G::slot(*this,&Run::clientEvent) ) ;
}
Main::Run::~Run()
{
if( m_store.get() ) m_store->signal().disconnect() ;
if( m_smtp_server.get() ) m_smtp_server->eventSignal().disconnect() ;
m_client.doneSignal().disconnect() ;
m_client.eventSignal().disconnect() ;
// avoid 'still reachable' in valgrind leak checks
m_client.reset() ;
m_poll_timer <<= 0 ;
m_forwarding_timer <<= 0 ;
m_admin_server <<= 0 ;
m_pop_secrets <<= 0 ;
m_client_secrets <<= 0 ;
m_store <<= 0 ;
m_log_output <<= 0 ;
m_cl <<= 0 ;
}
bool Main::Run::prepareError() const
{
return m_prepare_error ;
}
const Main::Configuration & Main::Run::config() const
{
cl() ;
return *m_cfg.get() ;
}
std::string Main::Run::smtpIdent() const
{
return std::string("E-MailRelay V") + versionNumber() ;
}
void Main::Run::closeFiles()
{
if( config().daemon() )
{
const bool keep_stderr = true ;
G::Process::closeFiles( keep_stderr ) ;
}
}
void Main::Run::closeMoreFiles()
{
if( config().closeStderr() )
G::Process::closeStderr() ;
}
bool Main::Run::hidden() const
{
return cl().contains("hidden") ;
}
bool Main::Run::prepare()
{
#ifndef USE_SMALL_CONFIG
if( cl().contains("help") )
{
cl().showHelp( false ) ;
m_prepare_error = false ;
return false ;
}
else if( cl().hasUsageErrors() )
{
cl().showUsageErrors( true ) ;
m_prepare_error = true ;
return false ;
}
else if( cl().contains("version") )
{
cl().showVersion( false ) ;
m_prepare_error = false ;
return false ;
}
else if( cl().argc() > 1U )
{
cl().showArgcError( true ) ;
m_prepare_error = true ;
return false ;
}
else if( cl().hasSemanticError() )
{
cl().showSemanticError( true ) ;
m_prepare_error = true ;
return false ;
}
#endif
// early singletons...
//
const Configuration & cfg = config() ;
m_log_output <<= new G::LogOutput( m_arg.prefix() ,
cfg.log() , // output
cfg.log() , // with-logging
cfg.verbose() , // with-verbose-logging
cfg.debug() , // with-debug
true , // with-level
cfg.logTimestamp() , // with-timestamp
!cfg.debug() , // strip-context
cfg.useSyslog() , // use-syslog
G::LogOutput::Mail // facility
) ;
#ifndef USE_SMALL_CONFIG
// emit warnings for dodgy switch combinations
cl().logSemanticWarnings() ;
#endif
return true ;
}
void Main::Run::run()
{
try
{
runCore() ;
G_LOG( "Main::Run::run: done" ) ;
}
catch( std::exception & e )
{
G_ERROR( "Main::Run::run: " << e.what() ) ;
throw ;
}
catch(...)
{
G_ERROR( "Main::Run::run: unknown exception" ) ;
throw ;
}
}
#ifndef USE_SMALL_CONFIG
void Main::Run::checkPort( const std::string & interface_ , unsigned int port )
{
GNet::Address address =
interface_.length() ?
GNet::Address(interface_,port) :
GNet::Address(port) ;
GNet::Server::canBind( address , true ) ;
}
#endif
void Main::Run::checkPorts() const
{
#ifndef USE_SMALL_CONFIG
const Configuration & cfg = config() ;
G::Strings interfaces = cfg.listeningInterfaces() ;
for( G::Strings::iterator p = interfaces.begin() ; p != interfaces.end() ; ++p )
{
if( cfg.doServing() && cfg.doSmtp() )
checkPort( *p , cfg.port() ) ;
if( cfg.doServing() && cfg.doPop() )
checkPort( *p , cfg.popPort() ) ;
if( cfg.doServing() && cfg.doAdmin() )
checkPort( *p , cfg.adminPort() ) ;
}
#endif
}
void Main::Run::runCore()
{
const Configuration & cfg = config() ;
// fqdn override option
//
GNet::Local::fqdn( cfg.fqdn() ) ;
// tighten the umask
//
G::Process::Umask::set( G::Process::Umask::Tightest ) ;
// close inherited file descriptors to avoid locking file
// systems when running as a daemon -- this has to be done
// early, before opening any sockets or message-store streams
//
if( cfg.daemon() )
{
closeFiles() ;
}
// release root privileges and extra group memberships
//
G::Root::init( cfg.nobody() ) ;
// event loop singletons
//
GNet::TimerList timer_list ;
std::auto_ptr<GNet::EventLoop> event_loop(GNet::EventLoop::create()) ;
if( ! event_loop->init() )
throw G::Exception( "cannot initialise network layer" ) ;
// early check on socket bindability
//
checkPorts() ;
#ifndef USE_NO_EXEC
// early check on script executablity
//
checkScripts() ;
#endif
// ssl library singleton
//
bool ssl_active = cfg.clientTls() || !cfg.serverTlsFile().empty() ;
GSsl::Library ssl( ssl_active , cfg.serverTlsFile() ) ;
if( ssl_active && !ssl.enabled() )
throw G::Exception( "cannot do tls/ssl: openssl not built in" ) ;
// network monitor singleton
//
GNet::Monitor monitor ;
monitor.signal().connect( G::slot(*this,&Run::raiseNetworkEvent) ) ;
// message store singletons
//
m_store <<= new GSmtp::FileStore( cfg.spoolDir() , false , cfg.maxSize() ) ;
m_store->signal().connect( G::slot(*this,&Run::raiseStoreEvent) ) ;
GPop::Store pop_store( cfg.spoolDir() , cfg.popByName() , ! cfg.popNoDelete() ) ;
// authentication secrets
//
m_client_secrets <<= new GSmtp::Secrets( cfg.clientSecretsFile() , "client" ) ;
GSmtp::Secrets server_secrets( cfg.serverSecretsFile() , "server" ) ;
if( cfg.doPop() )
m_pop_secrets <<= new GPop::Secrets( cfg.popSecretsFile() ) ;
// daemonise
//
G::PidFile pid_file ;
if( cfg.usePidFile() ) pid_file.init( G::Path(cfg.pidFile()) ) ;
if( cfg.daemon() ) G::Daemon::detach( pid_file ) ;
// run as forwarding agent
//
if( cfg.doForwardingOnStartup() )
{
if( m_store->empty() )
cl().showNoop( true ) ;
else
doForwardingOnStartup( *m_store.get() , *m_client_secrets.get() , *event_loop.get() ) ;
}
// run as storage daemon
//
if( cfg.doServing() )
{
doServing( *m_client_secrets.get() , *m_store.get() , server_secrets ,
pop_store , *m_pop_secrets.get() , pid_file , *event_loop.get() ) ;
}
}
void Main::Run::doServing( const GSmtp::Secrets & client_secrets ,
GSmtp::MessageStore & store , const GSmtp::Secrets & server_secrets ,
GPop::Store & pop_store , const GPop::Secrets & pop_secrets ,
G::PidFile & pid_file , GNet::EventLoop & event_loop )
{
const Configuration & cfg = config() ;
if( cfg.doSmtp() )
{
if( cfg.immediate() )
G_WARNING( "Run::doServing: using --immediate can result in client timeout errors: try --poll=0 instead" ) ;
m_smtp_server <<= new GSmtp::Server(
store ,
client_secrets ,
server_secrets ,
serverConfig() ,
cfg.immediate() ? cfg.serverAddress() : std::string() ,
cfg.connectionTimeout() ,
clientConfig() ) ;
m_smtp_server->eventSignal().connect( G::slot(*this,&Run::serverEvent) ) ;
}
#ifndef USE_NO_POP
std::auto_ptr<GPop::Server> pop_server ;
if( cfg.doPop() )
{
pop_server <<= new GPop::Server( pop_store , pop_secrets , popConfig() ) ;
}
#endif
#ifndef USE_NO_ADMIN
if( cfg.doAdmin() )
{
std::auto_ptr<GSmtp::AdminServer> admin_server = Admin::newServer( cfg , store , clientConfig() ,
client_secrets , versionNumber() ) ;
m_admin_server <<= admin_server.release() ;
}
#endif
if( cfg.doPolling() )
{
m_poll_timer <<= new GNet::Timer<Run>(*this,&Run::onPollTimeout,*this) ;
m_poll_timer->startTimer( cfg.pollingTimeout() ) ;
}
if( cfg.forwardingOnStore() || cfg.forwardingOnDisconnect() )
{
m_forwarding_timer <<= new GNet::Timer<Run>(*this,&Run::onForwardingTimeout,*this) ;
}
{
// dont change the effective group id here -- create the pid file with the
// unprivileged group ownership so that it can be deleted more easily
G::Root claim_root(false) ;
pid_file.commit() ;
}
closeMoreFiles() ;
if( m_smtp_server.get() ) m_smtp_server->report() ;
if( m_admin_server.get() ) Admin::report( *m_admin_server.get() ) ;
#ifndef USE_NO_POP
if( pop_server.get() ) pop_server->report() ;
#endif
event_loop.run() ;
if( m_smtp_server.get() ) m_smtp_server->eventSignal().disconnect() ;
m_smtp_server <<= 0 ;
m_admin_server <<= 0 ;
}
void Main::Run::doForwardingOnStartup( GSmtp::MessageStore & store , const GSmtp::Secrets & secrets ,
GNet::EventLoop & event_loop )
{
const Configuration & cfg = config() ;
GNet::ClientPtr<GSmtp::Client> client_ptr( new GSmtp::Client(
GNet::ResolverInfo(cfg.serverAddress()) , secrets , clientConfig() ) ) ;
client_ptr->sendMessages( store ) ;
client_ptr.doneSignal().connect( G::slot(*this,&Run::forwardingClientDone) ) ;
client_ptr.eventSignal().connect( G::slot(*this,&Run::clientEvent) ) ;
closeMoreFiles() ;
event_loop.run() ;
}
GSmtp::Server::Config Main::Run::serverConfig() const
{
const Configuration & cfg = config() ;
GSmtp::Server::AddressList interfaces ;
{
G::Strings list = cfg.listeningInterfaces() ;
for( G::Strings::iterator p = list.begin() ; p != list.end() ; ++p )
{
if( (*p).length() )
interfaces.push_back( GNet::Address(*p,cfg.port()) ) ;
}
}
return
GSmtp::Server::Config(
cfg.allowRemoteClients() ,
cfg.port() ,
interfaces ,
smtpIdent() ,
cfg.anonymous() ,
cfg.filter() ,
cfg.filterTimeout() ,
cfg.verifier() ,
cfg.filterTimeout() ) ; // verifier timeout
}
#ifndef USE_NO_POP
GPop::Server::Config Main::Run::popConfig() const
{
const Configuration & cfg = config() ;
return GPop::Server::Config( cfg.allowRemoteClients() , cfg.popPort() , cfg.listeningInterfaces() ) ;
}
#endif
GSmtp::Client::Config Main::Run::clientConfig() const
{
const Configuration & cfg = config() ;
return
GSmtp::Client::Config(
cfg.clientFilter() ,
cfg.filterTimeout() ,
cfg.clientInterface().length() ?
GNet::Address(cfg.clientInterface(),0U) :
GNet::Address(0U) ,
GSmtp::ClientProtocol::Config(
GNet::Local::fqdn() ,
cfg.responseTimeout() ,
cfg.promptTimeout() , // waiting for "service ready"
cfg.filterTimeout() ,
true , // (must-authenticate)
false ) , // (eight-bit-strict)
cfg.connectionTimeout() ) ;
}
void Main::Run::onException( std::exception & e )
{
// gets here if onTimeout() throws
G_ERROR( "Main::Run::onException: exception while forwarding: " << e.what() ) ;
}
void Main::Run::onPollTimeout()
{
G_DEBUG( "Main::Run::onPollTimeout" ) ;
m_poll_timer->startTimer( config().pollingTimeout() ) ;
doForwarding( "poll" ) ;
}
void Main::Run::onForwardingTimeout()
{
G_DEBUG( "Main::Run::onForwardingTimeout" ) ;
doForwarding( "forward" ) ;
}
void Main::Run::doForwarding( const std::string & event_key )
{
if( m_client.busy() )
{
G_DEBUG( "Main::Run::doForwarding: still busy from last time" ) ;
emit( event_key , "busy" , "" ) ;
}
else
{
emit( event_key , "start" , "" ) ;
std::string error = doForwardingCore() ;
emit( event_key , "end" , error ) ;
}
}
std::string Main::Run::doForwardingCore()
{
try
{
const Configuration & cfg = config() ;
G_DEBUG( "Main::Run::doForwarding: polling" ) ;
if( cfg.pollingLog() )
G_LOG( "Main::Run::doForwarding: polling" ) ;
if( ! m_store->empty() )
{
m_client.reset( new GSmtp::Client( GNet::ResolverInfo(cfg.serverAddress()) ,
*m_client_secrets.get() , clientConfig() ) ) ;
m_client->sendMessages( *m_store.get() ) ;
}
return std::string() ;
}
catch( std::exception & e )
{
G_ERROR( "Main::Run::doForwarding: polling: " << e.what() ) ;
return e.what() ;
}
}
const Main::CommandLine & Main::Run::cl() const
{
// lazy evaluation so that the constructor doesnt throw
if( m_cl.get() == NULL )
{
const_cast<Run*>(this)->m_cl <<= new CommandLine( m_output , m_arg , m_switch_spec , versionNumber() ) ;
const_cast<Run*>(this)->m_cfg <<= new Configuration( cl().cfg() ) ;
}
return *m_cl.get() ;
}
void Main::Run::forwardingClientDone( std::string reason , bool )
{
G_DEBUG( "Main::Run::forwardingClientDone: \"" << reason << "\"" ) ;
if( ! reason.empty() )
throw G::Exception( reason ) ;
else
GNet::EventLoop::instance().quit() ;
}
void Main::Run::pollingClientDone( std::string reason , bool )
{
G_DEBUG( "Main::Run::pollingClientDone: \"" << reason << "\"" ) ;
if( ! reason.empty() )
{
G_ERROR( "Main::Run::pollingClientDone: polling: " << reason ) ;
}
}
void Main::Run::clientEvent( std::string s1 , std::string s2 )
{
emit( "client" , s1 , s2 ) ;
}
void Main::Run::serverEvent( std::string s1 , std::string )
{
if( s1 == "done" && config().forwardingOnDisconnect() )
{
G_ASSERT( m_forwarding_timer.get() ) ;
m_forwarding_timer->cancelTimer() ;
m_forwarding_timer->startTimer( 0U ) ;
}
}
void Main::Run::raiseStoreEvent( bool repoll )
{
emit( "store" , "update" , repoll ? std::string("poll") : std::string() ) ;
const bool expiry_forced_by_filter = repoll ;
if( config().doPolling() && expiry_forced_by_filter )
{
G_ASSERT( m_poll_timer.get() ) ;
m_poll_timer->cancelTimer() ;
m_poll_timer->startTimer( 0U ) ;
}
if( config().forwardingOnStore() )
{
G_ASSERT( m_forwarding_timer.get() ) ;
m_forwarding_timer->cancelTimer() ;
m_forwarding_timer->startTimer( 0U ) ;
}
}
void Main::Run::raiseNetworkEvent( std::string s1 , std::string s2 )
{
emit( "network" , s1 , s2 ) ;
}
void Main::Run::emit( const std::string & s0 , const std::string & s1 , const std::string & s2 )
{
#ifndef USE_NO_ADMIN
try
{
m_signal.emit( s0 , s1 , s2 ) ;
if( m_admin_server.get() != NULL )
{
Admin::notify( *m_admin_server.get() , s0 , s1 , s2 ) ;
}
}
catch( std::exception & e )
{
G_WARNING( "Main::Run::emit: " << e.what() ) ;
}
#endif
}
G::Signal3<std::string,std::string,std::string> & Main::Run::signal()
{
return m_signal ;
}
#ifndef USE_NO_EXEC
void Main::Run::checkScripts() const
{
const Configuration & cfg = config() ;
checkProcessorScript( cfg.filter() ) ;
checkProcessorScript( cfg.clientFilter() ) ;
checkVerifierScript( cfg.verifier() ) ;
}
void Main::Run::checkVerifierScript( const std::string & s ) const
{
std::string reason = GSmtp::VerifierFactory::check( s ) ;
if( !reason.empty() )
{
G_WARNING( "Main::Run::checkScript: invalid verifier \"" << s << "\": " << reason ) ;
}
}
void Main::Run::checkProcessorScript( const std::string & s ) const
{
std::string reason = GSmtp::ProcessorFactory::check( s ) ;
if( !reason.empty() )
{
G_WARNING( "Main::Run::checkScript: invalid preprocessor \"" << s << "\": " << reason ) ;
}
}
#endif
/// \file run.cpp