emailrelay/src/gnet/gclient.cpp
Graeme Walker 1f92260cdf v0.9.5
2001-10-27 12:00:00 +00:00

500 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.
//
// ===
//
// gclient.cpp
//
#include "gdef.h"
#include "gnet.h"
#include "gaddress.h"
#include "gsocket.h"
#include "gdatetime.h"
#include "gresolve.h"
#include "gmonitor.h"
#include "gclient.h"
#include "gdebug.h"
#include "gassert.h"
#include "glog.h"
namespace
{
const int c_retries = 10 ; // number of retries when using a priviledged local port number
const int c_port_start = 512 ;
const int c_port_end = 1024 ;
} ;
namespace GNet
{
class ClientResolver ;
} ;
// Class: GNet::ClientResolver
// Description: A resolver class which calls ClientImp::resolveCon() when done.
//
class GNet::ClientResolver : public GNet:: Resolver
{
private:
ClientImp & m_client_imp ;
public:
ClientResolver( ClientImp & imp ) ;
void resolveCon( bool success ,
const Address &address , std::string reason ) ;
private:
ClientResolver( const ClientResolver & ) ;
void operator=( const ClientResolver & ) ;
} ;
inline
GNet::ClientResolver::ClientResolver( ClientImp & imp ) :
m_client_imp(imp)
{
}
// ===
// Class: GNet::ClientImp
// Description: A pimple-pattern implementation class for GClient.
//
class GNet::ClientImp : public GNet:: EventHandler
{
private:
ClientResolver m_resolver ;
StreamSocket * m_s ;
Address m_address ;
Client & m_interface ;
bool m_priviledged ;
enum Status { Success , Failure , Retry , ImmediateSuccess } ;
static bool m_first ;
enum State { Idle , Resolving , Connecting , Connected , Failed , Disconnected } ;
State m_state ;
bool m_quit_on_disconnect ;
public:
ClientImp( Client &intaface , bool priviledged , bool quit_on_disconnect ) ;
virtual ~ClientImp() ;
void resolveCon( bool ok , const Address & address , std::string reason ) ;
void readEvent() ;
void writeEvent() ;
void exceptionEvent() ;
bool connect( std::string host , std::string service , std::string *error , bool sync_dns ) ;
std::string startConnecting( const Address & , bool & ) ;
Status connectCore( Address , std::string * , bool , unsigned int ) ;
void disconnect() ;
StreamSocket & s() ;
const StreamSocket & s() const ;
void run() ;
void close() ;
void blocked() ;
bool connected() const ;
void setState( State ) ;
std::pair<bool,Address> localAddress() const ;
std::pair<bool,Address> peerAddress() const ;
private:
ClientImp( const ClientImp & ) ;
void operator=( const ClientImp & ) ;
static int getRandomPort() ;
} ;
// ===
GNet::Client::Client( bool priviledged , bool quit_on_disconnect ) :
m_imp(NULL)
{
G_DEBUG( "Client::ctor" ) ;
m_imp = new ClientImp( *this , priviledged , quit_on_disconnect ) ;
if( Monitor::instance() ) Monitor::instance()->add( *this ) ;
}
GNet::Client::~Client()
{
if( Monitor::instance() ) Monitor::instance()->remove( *this ) ;
delete m_imp ;
}
bool GNet::Client::connect( std::string host , std::string service , std::string *error_p , bool sync_dns )
{
return m_imp->connect( host , service , error_p , sync_dns ) ;
}
void GNet::Client::run()
{
m_imp->run() ;
}
bool GNet::Client::connected() const
{
return m_imp->connected() ;
}
void GNet::Client::blocked()
{
m_imp->blocked() ;
}
void GNet::Client::disconnect()
{
m_imp->disconnect() ;
}
std::pair<bool,GNet::Address> GNet::Client::localAddress() const
{
return m_imp->localAddress() ;
}
std::pair<bool,GNet::Address> GNet::Client::peerAddress() const
{
return m_imp->peerAddress() ;
}
// ===
bool GNet::ClientImp::m_first = true ;
GNet::ClientImp::ClientImp( Client &intaface , bool priviledged , bool quit_on_disconnect ) :
m_interface(intaface) ,
m_state(Idle) ,
m_s(NULL) ,
m_resolver(*this) ,
m_priviledged(priviledged) ,
m_address(Address::invalidAddress()) ,
m_quit_on_disconnect(quit_on_disconnect)
{
G_DEBUG( "ClientImp::ctor" ) ;
}
int GNet::ClientImp::getRandomPort()
{
if( m_first )
{
std::srand( static_cast<unsigned int>(G::DateTime::now()) ) ;
m_first = false ;
}
int r = std::rand() ;
if( r < 0 ) r = -r ;
r = r % (c_port_end - c_port_start) ;
return r + c_port_start ;
}
GNet::StreamSocket & GNet::ClientImp::s()
{
G_ASSERT( m_s != NULL ) ;
return *m_s ;
}
const GNet::StreamSocket & GNet::ClientImp::s() const
{
G_ASSERT( m_s != NULL ) ;
return *m_s ;
}
GNet::ClientImp::~ClientImp()
{
setState( Disconnected ) ; // for quit()
close() ;
}
void GNet::ClientImp::disconnect()
{
setState( Disconnected ) ;
close() ;
}
void GNet::ClientImp::close()
{
delete m_s ;
m_s = NULL ;
}
bool GNet::ClientImp::connected() const
{
return m_state == Connected ;
}
bool GNet::ClientImp::connect( std::string host , std::string service ,
std::string *error_p , bool sync_dns )
{
G_DEBUG( "GNet::ClientImp::connect: \"" << host << "\", \"" << service << "\"" ) ;
std::string dummy_error_string ;
if( error_p == NULL )
error_p = &dummy_error_string ;
std::string &error = *error_p ;
if( sync_dns )
{
std::pair<Resolver::HostInfo,std::string> pair = Resolver::resolve( host , service ) ;
std::string & resolve_reason = pair.second ;
if( resolve_reason.length() != 0U )
{
error = resolve_reason ;
setState( Failed ) ;
return false ;
}
bool immediate = false ;
std::string connect_reason = startConnecting( pair.first.address , immediate ) ;
if( connect_reason.length() != 0U )
{
error = connect_reason ;
setState( Failed ) ;
return false ;
}
if( immediate )
{
G_WARNING( "GNet::Client::connect: immediate connection" ) ; // delete soon
s().addReadHandler( *this ) ;
s().addExceptionHandler( *this ) ;
setState( Connected ) ;
m_interface.onConnect( s() ) ; // from within connect() ?
}
else
{
setState( Connecting ) ;
}
}
else
{
std::string address_string( host ) ;
address_string.append( ":" ) ;
address_string.append( service.c_str() ) ;
if( !m_resolver.resolveReq( address_string ) )
{
error = "invalid host/service: " ;
error.append( address_string.c_str() ) ;
setState( Failed ) ;
return false ;
}
setState( Resolving ) ;
}
return true ;
}
void GNet::ClientImp::resolveCon( bool success , const Address &address ,
std::string resolve_reason )
{
if( success )
{
G_DEBUG( "GNet::ClientImp::resolveCon: " << address.displayString() ) ;
bool immediate = false ;
std::string connect_reason = startConnecting( address , immediate ) ;
if( connect_reason.length() )
{
close() ;
setState( Failed ) ;
m_interface.onError( connect_reason ) ;
}
setState( immediate ? Connected : Connecting ) ;
}
else
{
resolve_reason = std::string("resolver error: ") + resolve_reason ;
close() ;
setState( Failed ) ;
m_interface.onError( resolve_reason ) ;
}
}
std::string GNet::ClientImp::startConnecting( const Address & address , bool & immediate )
{
// save the target address
G_DEBUG( "GNet::ClientImp::startConnecting: " << address.displayString() ) ;
m_address = address ;
// create and open a socket
//
m_s = new StreamSocket ;
if( !s().valid() )
{
return std::string( "error: cannot open socket" ) ;
}
// specifiy this as a 'write' event handler for the socket
// (before the connect() in case it is reentrant)
//
s().addWriteHandler( *this ) ;
Status status = Failure ;
std::string error ;
if( m_priviledged )
{
for( int i = 0 ; i < c_retries ; i++ )
{
int port = getRandomPort() ;
G_DEBUG( "GNet::ClientImp::resolveCon: trying to bind port " << port ) ;
status = connectCore( address, &error, true, port ) ;
if( status != Retry )
break ;
}
}
else
{
status = connectCore( address , &error , false , 0 ) ;
}
immediate = status == ImmediateSuccess ;
if( status != Success )
s().dropWriteHandler() ;
if( status == Success ) error = std::string() ;
return error ;
}
GNet::ClientImp::Status GNet::ClientImp::connectCore( Address remote_address ,
std::string *error_p , bool set_port , unsigned int port )
{
G_ASSERT( error_p != NULL ) ;
std::string &error = *error_p ;
Address local_address( set_port ? port : 0 ) ;
bool bound = s().bind(local_address) ;
if( !bound )
{
error = "error: cannot bind socket" ;
return Retry ;
}
G_DEBUG( "GNet::ClientImp::connectCore: bound local address "
<< local_address.displayString() ) ;
// initiate the connection
//
bool immediate = false ;
if( !s().connect( remote_address , &immediate ) )
{
G_DEBUG( "GNet::ClientImp::connect: immediate failure" ) ;
error = "error: cannot connect to " ;
error.append( remote_address.displayString().c_str() ) ;
// we should return Failure here, but Microsoft's stack
// will happily bind the same local address more than once,
// so it is the connect that fails, not the bind, if
// the port was already in use
//
return Retry ;
}
else
{
return immediate ? ImmediateSuccess : Success ;
}
}
void GNet::ClientImp::blocked()
{
s().addWriteHandler( *this ) ;
}
void GNet::ClientImp::writeEvent()
{
G_DEBUG( "GNet::ClientImp::writeEvent" ) ;
if( m_state == Connected )
{
s().dropWriteHandler() ;
m_interface.onWriteable() ;
}
else if( m_state == Connecting && s().hasPeer() )
{
s().addReadHandler( *this ) ;
s().addExceptionHandler( *this ) ;
s().dropWriteHandler() ;
setState( Connected ) ;
m_interface.onConnect( s() ) ;
}
else if( m_state == Connecting )
{
std::string message( "error: cannot connect to " ) ;
message.append( m_address.displayString().c_str() ) ;
setState( Failed ) ;
close() ;
m_interface.onError( message ) ;
}
}
void GNet::ClientImp::readEvent()
{
char buffer[200U] ;
ssize_t n = s().read( buffer , sizeof(buffer) ) ;
if( n == 0 || ( n == -1 && !s().eWouldBlock() ) )
{
close() ;
setState( Disconnected ) ;
m_interface.onDisconnect() ;
}
else if( n != -1 )
{
G_ASSERT( n <= sizeof(buffer) ) ;
G_DEBUG( "GNet::ClientImp::readEvent: " << n << " byte(s)" ) ;
m_interface.onData( buffer , n ) ;
}
else
{
; // no-op (windows)
}
}
void GNet::ClientImp::exceptionEvent()
{
G_DEBUG( "GNet::ClientImp::exceptionEvent" ) ;
close() ;
setState( Failed ) ;
m_interface.onDisconnect() ;
}
void GNet::ClientImp::setState( State new_state )
{
if( m_quit_on_disconnect &&
(new_state == Disconnected || new_state == Failed) &&
(m_state != Disconnected && m_state != Failed) )
{
G_DEBUG( "GNet::ClientImp::setState: " << m_state
<< " -> " << new_state << ": quitting the event loop" ) ;
EventSources::instance().quit() ;
}
m_state = new_state ;
}
void GNet::ClientImp::run()
{
EventSources::instance().run() ;
}
std::pair<bool,GNet::Address> GNet::ClientImp::localAddress() const
{
return s().getLocalAddress() ;
}
std::pair<bool,GNet::Address> GNet::ClientImp::peerAddress() const
{
return s().getPeerAddress() ;
}
// ===
void GNet::ClientResolver::resolveCon( bool success , const Address &address ,
std::string reason )
{
m_client_imp.resolveCon( success , address , reason ) ;
}