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

512 lines
13 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/>.
// ===
//
// gsocketprotocol.cpp
//
#include "gdef.h"
#include "glimits.h"
#include "gnet.h"
#include "gssl.h"
#include "gsocketprotocol.h"
#include "gtest.h"
#include "gassert.h"
#include "glog.h"
namespace
{
const size_t c_buffer_size = G::limits::net_buffer ;
}
/// \class GNet::SocketProtocolImp
/// A private implementation class used by GNet::SocketProtocol.
///
class GNet::SocketProtocolImp
{
private:
enum State { State_raw , State_connecting , State_accepting , State_writing , State_idle } ;
EventHandler & m_handler ;
SocketProtocol::Sink & m_sink ;
StreamSocket & m_socket ;
const Socket::Credentials & m_credentials ;
std::string m_raw_residue ;
std::string m_ssl_send_data ;
bool m_failed ;
unsigned long m_n ;
GSsl::Protocol * m_ssl ;
State m_state ;
char m_read_buffer[c_buffer_size] ;
GSsl::Protocol::ssize_type m_read_buffer_size ;
GSsl::Protocol::ssize_type m_read_buffer_n ;
public:
SocketProtocolImp( EventHandler & , SocketProtocol::Sink & , StreamSocket & , const Socket::Credentials & ) ;
~SocketProtocolImp() ;
void readEvent() ;
bool writeEvent() ;
bool send( const std::string & data , std::string::size_type offset ) ;
void sslConnect() ;
void sslAccept() ;
bool sslEnabled() const ;
private:
SocketProtocolImp( const SocketProtocolImp & ) ;
void operator=( const SocketProtocolImp & ) ;
static GSsl::Protocol * newProtocol() ;
static void log( int level , const std::string & line ) ;
StreamSocket & socket() ;
bool failed() const ;
bool rawSendImp( const std::string & , std::string::size_type , std::string & ) ;
void rawReadEvent() ;
bool rawWriteEvent() ;
bool rawSend( const std::string & data , std::string::size_type offset ) ;
void sslReadImp() ;
bool sslSendImp() ;
void sslConnectImp() ;
void sslAcceptImp() ;
void logFlowControlReleased() ;
void logFlowControlAsserted() ;
void logFlowControlReasserted() ;
} ;
GNet::SocketProtocolImp::SocketProtocolImp( EventHandler & handler ,
SocketProtocol::Sink & sink , StreamSocket & socket , const Socket::Credentials & credentials ) :
m_handler(handler) ,
m_sink(sink) ,
m_socket(socket) ,
m_credentials(credentials) ,
m_failed(false) ,
m_n(0UL) ,
m_ssl(NULL) ,
m_state(State_raw) ,
m_read_buffer_size(sizeof(m_read_buffer)) ,
m_read_buffer_n(0)
{
}
GNet::SocketProtocolImp::~SocketProtocolImp()
{
delete m_ssl ;
}
GNet::StreamSocket & GNet::SocketProtocolImp::socket()
{
return m_socket ;
}
void GNet::SocketProtocolImp::readEvent()
{
G_DEBUG( "SocketProtocolImp::readEvent: state=" << m_state ) ;
if( m_state == State_raw )
rawReadEvent() ;
else if( m_state == State_connecting )
sslConnectImp() ;
else if( m_state == State_accepting )
sslAcceptImp() ;
else if( m_state == State_writing )
sslSendImp() ;
else // State_idle
sslReadImp() ;
}
bool GNet::SocketProtocolImp::writeEvent()
{
G_DEBUG( "GNet::SocketProtocolImp::writeEvent: state=" << m_state ) ;
bool rc = true ;
if( m_state == State_raw )
rc = rawWriteEvent() ;
else if( m_state == State_connecting )
sslConnectImp() ;
else if( m_state == State_accepting )
sslAcceptImp() ;
else
sslSendImp() ;
return rc ;
}
bool GNet::SocketProtocolImp::send( const std::string & data , std::string::size_type offset )
{
if( data.empty() || offset >= data.length() )
return true ;
bool rc = true ;
if( m_state == State_raw )
{
rc = rawSend( data , offset ) ;
}
else if( m_state == State_connecting || m_state == State_accepting )
{
throw SocketProtocol::SendError( "still busy negotiating" ) ;
}
else if( m_state == State_writing )
{
// throw here rather than add to the pending buffer because openssl
// requires that the parameters stay the same -- we could use double
// buffering, with a buffer switch and a call to sslSendImp()
// rather than returning to the idle state, but in practice
// we rely on the client code taking account of the return value
// from send() and waiting for onSendComplete() when required
//
throw SocketProtocol::SendError( "still busy sending the last packet" ) ;
}
else
{
m_state = State_writing ;
m_ssl_send_data.append( data.substr(offset) ) ;
rc = sslSendImp() ;
}
return rc ;
}
void GNet::SocketProtocolImp::log( int level , const std::string & log_line )
{
if( level == 0 )
G_DEBUG( "ssl: " << log_line ) ;
else if( level == 1 )
G_DEBUG( "SocketProtocolImp::log: " << log_line ) ;
else
G_WARNING( "GNet::SocketProtocolImp::log: " << log_line ) ;
}
GSsl::Protocol * GNet::SocketProtocolImp::newProtocol()
{
GSsl::Library * library = GSsl::Library::instance() ;
if( library == NULL )
throw G::Exception( "SocketProtocolImp::newProtocol: internal error: no library instance" ) ;
return new GSsl::Protocol( *library , log ) ;
}
void GNet::SocketProtocolImp::sslConnect()
{
G_DEBUG( "SocketProtocolImp::sslConnect" ) ;
G_ASSERT( m_ssl == NULL ) ;
m_ssl = newProtocol() ;
m_state = State_connecting ;
sslConnectImp() ;
}
void GNet::SocketProtocolImp::sslConnectImp()
{
G_DEBUG( "SocketProtocolImp::sslConnectImp" ) ;
G_ASSERT( m_ssl != NULL ) ;
G_ASSERT( m_state == State_connecting ) ;
GSsl::Protocol::Result rc = m_ssl->connect( m_socket.fd(m_credentials) ) ;
G_DEBUG( "SocketProtocolImp::sslConnectImp: result=" << GSsl::Protocol::str(rc) ) ;
if( rc == GSsl::Protocol::Result_error )
{
socket().dropWriteHandler() ;
m_state = State_raw ;
throw SocketProtocol::ReadError( "ssl connect" ) ;
}
else if( rc == GSsl::Protocol::Result_read )
{
socket().dropWriteHandler() ;
}
else if( rc == GSsl::Protocol::Result_write )
{
socket().addWriteHandler( m_handler ) ;
}
else
{
socket().dropWriteHandler() ;
m_state = State_idle ;
G_DEBUG( "SocketProtocolImp::sslConnectImp: calling onSecure" ) ;
m_sink.onSecure() ;
}
}
void GNet::SocketProtocolImp::sslAccept()
{
G_DEBUG( "SocketProtocolImp::sslAccept" ) ;
G_ASSERT( m_ssl == NULL ) ;
m_ssl = newProtocol() ;
m_state = State_accepting ;
sslAcceptImp() ;
}
void GNet::SocketProtocolImp::sslAcceptImp()
{
G_DEBUG( "SocketProtocolImp::sslAcceptImp" ) ;
G_ASSERT( m_ssl != NULL ) ;
G_ASSERT( m_state == State_accepting ) ;
GSsl::Protocol::Result rc = m_ssl->accept( m_socket.fd(m_credentials) ) ;
G_DEBUG( "SocketProtocolImp::sslAcceptImp: result=" << GSsl::Protocol::str(rc) ) ;
if( rc == GSsl::Protocol::Result_error )
{
socket().dropWriteHandler() ;
m_state = State_raw ;
throw SocketProtocol::ReadError( "ssl accept" ) ;
}
else if( rc == GSsl::Protocol::Result_read )
{
socket().dropWriteHandler() ;
}
else if( rc == GSsl::Protocol::Result_write )
{
socket().addWriteHandler( m_handler ) ;
}
else
{
socket().dropWriteHandler() ;
m_state = State_idle ;
G_DEBUG( "SocketProtocolImp::sslAcceptImp: calling onSecure" ) ;
m_sink.onSecure() ;
}
}
bool GNet::SocketProtocolImp::sslEnabled() const
{
return m_state == State_writing || m_state == State_idle ;
}
bool GNet::SocketProtocolImp::sslSendImp()
{
G_ASSERT( m_state == State_writing ) ;
bool rc = false ;
GSsl::Protocol::ssize_type n = 0 ;
GSsl::Protocol::Result result = m_ssl->write( m_ssl_send_data.data() , m_ssl_send_data.size() , n ) ;
if( result == GSsl::Protocol::Result_error )
{
socket().dropWriteHandler() ;
m_state = State_idle ;
throw SocketProtocol::SendError( "ssl write" ) ;
}
else if( result == GSsl::Protocol::Result_read )
{
socket().dropWriteHandler() ;
}
else if( result == GSsl::Protocol::Result_write )
{
socket().addWriteHandler( m_handler ) ;
}
else
{
socket().dropWriteHandler() ;
rc = n == static_cast<GSsl::Protocol::ssize_type>(m_ssl_send_data.size()) ;
m_ssl_send_data.erase( 0U , n ) ;
m_state = State_idle ;
}
return rc ;
}
void GNet::SocketProtocolImp::sslReadImp()
{
G_DEBUG( "SocketProtocolImp::sslReadImp" ) ;
G_ASSERT( m_state == State_idle ) ;
G_ASSERT( m_ssl != NULL ) ;
for( int sanity = 0 ; sanity < 1000 ; sanity++ )
{
GSsl::Protocol::Result rc = m_ssl->read( m_read_buffer , m_read_buffer_size , m_read_buffer_n ) ;
G_DEBUG( "SocketProtocolImp::sslReadImp: result=" << GSsl::Protocol::str(rc) ) ;
if( rc == GSsl::Protocol::Result_error )
{
socket().dropWriteHandler() ;
m_state = State_idle ;
throw SocketProtocol::ReadError( "ssl read" ) ;
}
else if( rc == GSsl::Protocol::Result_read )
{
socket().dropWriteHandler() ;
}
else if( rc == GSsl::Protocol::Result_write )
{
socket().addWriteHandler( m_handler ) ;
}
else // Result_ok, Result_more
{
socket().dropWriteHandler() ;
m_state = State_idle ;
GSsl::Protocol::ssize_type n = m_read_buffer_n ;
m_read_buffer_n = 0 ;
G_DEBUG( "SocketProtocolImp::sslReadImp: calling onData(): " << n ) ;
m_sink.onData( m_read_buffer , static_cast<std::string::size_type>(n) ) ;
}
if( rc == GSsl::Protocol::Result_more )
G_DEBUG( "SocketProtocolImp::sslReadImp: more available to read from the ssl layer without i/o" ) ;
else
break ;
}
}
void GNet::SocketProtocolImp::rawReadEvent()
{
char buffer[c_buffer_size] ;
buffer[0] = '\0' ;
const size_t buffer_size = G::Test::enabled("small-client-input-buffer") ? 3 : sizeof(buffer) ;
ssize_t rc = socket().read( buffer , buffer_size ) ;
if( rc == 0 || ( rc == -1 && !socket().eWouldBlock() ) )
{
throw SocketProtocol::ReadError() ;
}
else if( rc != -1 )
{
G_ASSERT( static_cast<size_t>(rc) <= buffer_size ) ;
m_sink.onData( buffer , static_cast<std::string::size_type>(rc) ) ;
}
else
{
; // -1 && eWouldBlock() -- no-op (esp. for windows)
}
}
bool GNet::SocketProtocolImp::rawSend( const std::string & data , std::string::size_type offset )
{
bool all_sent = rawSendImp( data , offset , m_raw_residue ) ;
if( !all_sent && failed() )
throw SocketProtocol::SendError() ;
if( !all_sent )
{
socket().addWriteHandler( m_handler ) ;
logFlowControlAsserted() ;
}
return all_sent ;
}
bool GNet::SocketProtocolImp::rawWriteEvent()
{
socket().dropWriteHandler() ;
logFlowControlReleased() ;
bool all_sent = rawSendImp( m_raw_residue , 0 , m_raw_residue ) ;
if( !all_sent && failed() )
throw SocketProtocol::SendError() ;
if( !all_sent )
{
socket().addWriteHandler( m_handler ) ;
logFlowControlReasserted() ;
}
return all_sent ;
}
bool GNet::SocketProtocolImp::rawSendImp( const std::string & data , std::string::size_type offset ,
std::string & residue )
{
if( data.length() <= offset )
return true ; // nothing to do
ssize_t rc = socket().write( data.data()+offset , data.length()-offset ) ;
if( rc < 0 && ! socket().eWouldBlock() )
{
// fatal error, eg. disconnection
m_failed = true ;
residue.erase() ;
return false ;
}
else if( rc < 0 || static_cast<std::string::size_type>(rc) < (data.length()-offset) )
{
// flow control asserted
std::string::size_type sent = rc > 0 ? static_cast<size_t>(rc) : 0U ;
m_n += sent ;
residue = data ;
if( (sent+offset) != 0U )
residue.erase( 0U , sent+offset ) ;
return false ;
}
else
{
// all sent
m_n += data.length() ;
residue.erase() ;
return true ;
}
}
bool GNet::SocketProtocolImp::failed() const
{
return m_failed ;
}
void GNet::SocketProtocolImp::logFlowControlAsserted()
{
const bool log = G::Test::enabled("log-flow-control") ;
if( log )
G_LOG( "GNet::SocketProtocolImp::send: @" << socket().asString() << ": flow control asserted" ) ;
}
void GNet::SocketProtocolImp::logFlowControlReleased()
{
const bool log = G::Test::enabled("log-flow-control") ;
if( log )
G_LOG( "GNet::SocketProtocolImp::send: @" << socket().asString() << ": flow control released" ) ;
}
void GNet::SocketProtocolImp::logFlowControlReasserted()
{
const bool log = G::Test::enabled("log-flow-control") ;
if( log )
G_LOG( "GNet::SocketProtocolImp::send: @" << socket().asString() << ": flow control reasserted" ) ;
}
//
GNet::SocketProtocol::SocketProtocol( EventHandler & handler , Sink & sink , StreamSocket & socket ) :
m_imp( new SocketProtocolImp(handler,sink,socket,Socket::Credentials("")) )
{
}
GNet::SocketProtocol::~SocketProtocol()
{
delete m_imp ;
}
void GNet::SocketProtocol::readEvent()
{
m_imp->readEvent() ;
}
bool GNet::SocketProtocol::writeEvent()
{
return m_imp->writeEvent() ;
}
bool GNet::SocketProtocol::send( const std::string & data , std::string::size_type offset )
{
return m_imp->send( data , offset ) ;
}
bool GNet::SocketProtocol::sslCapable()
{
return GSsl::Library::instance() != NULL && GSsl::Library::instance()->enabled() ;
}
void GNet::SocketProtocol::sslConnect()
{
m_imp->sslConnect() ;
}
void GNet::SocketProtocol::sslAccept()
{
m_imp->sslAccept() ;
}
bool GNet::SocketProtocol::sslEnabled() const
{
return m_imp->sslEnabled() ;
}
//
GNet::SocketProtocolSink::~SocketProtocolSink()
{
}