243 lines
5.1 KiB
C++
243 lines
5.1 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.
|
|
//
|
|
// ===
|
|
//
|
|
// gsmtpclient.cpp
|
|
//
|
|
|
|
#include "gdef.h"
|
|
#include "gnet.h"
|
|
#include "gsmtp.h"
|
|
#include "glocal.h"
|
|
#include "gfile.h"
|
|
#include "gstr.h"
|
|
#include "gmemory.h"
|
|
#include "gsmtpclient.h"
|
|
#include "gresolve.h"
|
|
#include "gassert.h"
|
|
#include "glog.h"
|
|
|
|
//static
|
|
std::string GSmtp::Client::crlf()
|
|
{
|
|
return std::string("\015\012") ;
|
|
}
|
|
|
|
GSmtp::Client::Client( MessageStore & store , bool quit_on_disconnect ) :
|
|
GNet::Client(false,quit_on_disconnect) ,
|
|
m_callback(NULL) ,
|
|
m_store(&store) ,
|
|
m_buffer(crlf()) ,
|
|
m_protocol(*this,GNet::Local::fqdn()) ,
|
|
m_socket(NULL)
|
|
{
|
|
}
|
|
|
|
GSmtp::Client::Client( MessageStore & store , ClientCallback & callback , bool quit_on_disconnect ) :
|
|
GNet::Client(false,quit_on_disconnect) ,
|
|
m_callback(&callback) ,
|
|
m_store(&store) ,
|
|
m_buffer(crlf()) ,
|
|
m_protocol(*this,GNet::Local::fqdn()) ,
|
|
m_socket(NULL)
|
|
{
|
|
}
|
|
|
|
GSmtp::Client::Client( std::auto_ptr<StoredMessage> message , ClientCallback & callback ) :
|
|
GNet::Client(false,false) ,
|
|
m_callback(&callback) ,
|
|
m_store(NULL) ,
|
|
m_message(message) ,
|
|
m_buffer(crlf()) ,
|
|
m_protocol(*this,GNet::Local::fqdn()) ,
|
|
m_socket(NULL)
|
|
{
|
|
}
|
|
|
|
std::string GSmtp::Client::init( const std::string & s )
|
|
{
|
|
size_t pos = s.find(':') ;
|
|
if( pos == std::string::npos )
|
|
return "invalid address string: no colon (<host/ip>:<service/port>)" ;
|
|
|
|
return init( s.substr(0U,pos) , s.substr(pos+1U) ) ;
|
|
}
|
|
|
|
std::string GSmtp::Client::init( const std::string & host , const std::string & service )
|
|
{
|
|
if( m_store != NULL && m_store->empty() )
|
|
return "no messages to send" ;
|
|
|
|
std::string error ;
|
|
bool rc = connect( host , service , &error ) ;
|
|
return rc ? std::string() : error ;
|
|
}
|
|
|
|
bool GSmtp::Client::busy() const
|
|
{
|
|
return connected() ; // GNet::Client::connected()
|
|
}
|
|
|
|
bool GSmtp::Client::protocolSend( const std::string & line )
|
|
{
|
|
ssize_t rc = socket().write( line.data() , line.length() ) ;
|
|
if( rc < 0 )
|
|
{
|
|
m_pending = line ;
|
|
if( socket().eWouldBlock() )
|
|
blocked() ;
|
|
return false ;
|
|
}
|
|
else if( rc < line.length() )
|
|
{
|
|
m_pending = line.substr(rc) ;
|
|
blocked() ; // GNet::Client::blocked() => addWriteHandler()
|
|
return false ;
|
|
}
|
|
else
|
|
{
|
|
return true ;
|
|
}
|
|
}
|
|
|
|
void GSmtp::Client::onConnect( GNet::Socket & socket )
|
|
{
|
|
m_socket = &socket ;
|
|
if( m_store != NULL )
|
|
{
|
|
m_iter = m_store->iterator() ;
|
|
if( !sendNext() )
|
|
finish() ;
|
|
}
|
|
else
|
|
{
|
|
G_ASSERT( m_message.get() != NULL ) ;
|
|
start( *m_message.get() ) ;
|
|
}
|
|
}
|
|
|
|
bool GSmtp::Client::sendNext()
|
|
{
|
|
m_message <<= 0 ;
|
|
|
|
{
|
|
std::auto_ptr<StoredMessage> message( m_iter.next() ) ;
|
|
if( message.get() == NULL )
|
|
{
|
|
G_LOG( "GSmtp::Client: no more messages to send" ) ;
|
|
GNet::Socket * s = m_socket ;
|
|
m_socket = NULL ;
|
|
s->close() ;
|
|
return false ;
|
|
}
|
|
m_message = message ;
|
|
}
|
|
|
|
start( *m_message.get() ) ;
|
|
return true ;
|
|
}
|
|
|
|
void GSmtp::Client::start( StoredMessage & message )
|
|
{
|
|
m_protocol.start( message.from() , message.to() , message.eightBit() ,
|
|
message.extractContentStream() , *this ) ;
|
|
}
|
|
|
|
void GSmtp::Client::protocolDone( bool ok , const std::string & reason )
|
|
{
|
|
G_DEBUG( "GSmtp::Client::protocolDone: " << ok << ": " << reason ) ;
|
|
|
|
std::string error_message ;
|
|
if( !ok )
|
|
error_message = std::string("smtp client protocol failure: ") + reason ;
|
|
|
|
if( m_message.get() != NULL )
|
|
{
|
|
if( ok )
|
|
m_message->destroy() ;
|
|
else
|
|
m_message->fail( error_message ) ;
|
|
}
|
|
|
|
if( m_store == NULL || !sendNext() )
|
|
{
|
|
finish( error_message ) ;
|
|
}
|
|
}
|
|
|
|
void GSmtp::Client::onDisconnect()
|
|
{
|
|
doCallback( "connection to server lost" ) ;
|
|
m_socket = NULL ;
|
|
}
|
|
|
|
GNet::Socket & GSmtp::Client::socket()
|
|
{
|
|
if( m_socket == NULL )
|
|
throw NotConnected() ;
|
|
return * m_socket ;
|
|
}
|
|
|
|
void GSmtp::Client::onData( const char * data , size_t size )
|
|
{
|
|
m_buffer.add( std::string(data,size) ) ;
|
|
while( m_buffer.more() )
|
|
{
|
|
m_protocol.apply( m_buffer.line() ) ;
|
|
if( m_protocol.done() )
|
|
finish() ;
|
|
}
|
|
}
|
|
|
|
void GSmtp::Client::onError( const std::string & error )
|
|
{
|
|
doCallback( std::string("error on connection to server: ") + error ) ;
|
|
G_WARNING( "GSmtp::Client: error: \"" << error << "\"" ) ;
|
|
}
|
|
|
|
void GSmtp::Client::finish( const std::string & reason )
|
|
{
|
|
doCallback( reason ) ;
|
|
disconnect() ; // GNet::Client::disconnect()
|
|
}
|
|
|
|
void GSmtp::Client::doCallback( const std::string & reason )
|
|
{
|
|
if( m_callback != NULL )
|
|
{
|
|
m_callback->clientDone( reason ) ;
|
|
m_callback = NULL ;
|
|
}
|
|
}
|
|
|
|
void GSmtp::Client::onWriteable()
|
|
{
|
|
if( protocolSend(m_pending) )
|
|
{
|
|
m_protocol.sendDone() ;
|
|
}
|
|
}
|
|
|
|
// ===
|
|
|
|
GSmtp::Client::ClientCallback::~ClientCallback()
|
|
{
|
|
}
|
|
|