emailrelay/src/gsmtp/gadminserver.cpp
Graeme Walker 6798a91d8b v1.1.3
2003-11-03 12:00:00 +00:00

304 lines
7.3 KiB
C++

//
// Copyright (C) 2001-2003 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.
//
// ===
//
// gadminserver.cpp
//
#include "gdef.h"
#include "gnet.h"
#include "geventloop.h"
#include "gsmtp.h"
#include "gadminserver.h"
#include "gmessagestore.h"
#include "gstoredmessage.h"
#include "gmonitor.h"
#include "gslot.h"
#include "gstr.h"
#include "gmemory.h"
#include <algorithm> // std::find()
GSmtp::AdminPeer::AdminPeer( GNet::Server::PeerInfo peer_info , AdminServer & server ,
const std::string & server_address , bool with_terminate ) :
GNet::ServerPeer(peer_info) ,
m_buffer(crlf()) ,
m_server(server) ,
m_server_address(server_address) ,
m_notifying(false) ,
m_with_terminate(with_terminate)
{
G_LOG_S( "GSmtp::AdminPeer: admin connection from " << peer_info.m_address.displayString() ) ;
// dont prompt() here -- it confuses the poke program
}
GSmtp::AdminPeer::~AdminPeer()
{
// only safe because AdminServer::dtor calls serverCleanup() -- otherwise
// the derived part of the server may already be destroyed
m_server.unregister( this ) ;
if( m_client.get() ) m_client->doneSignal().disconnect() ;
}
void GSmtp::AdminPeer::clientDone( std::string s )
{
if( s.empty() )
send( "OK" ) ;
else
send( std::string("error: ") + s ) ;
prompt() ;
}
void GSmtp::AdminPeer::onDelete()
{
G_LOG_S( "GSmtp::AdminPeer: admin connection closed: " << peerAddress().second.displayString() ) ;
}
void GSmtp::AdminPeer::onData( const char * data , size_t n )
{
m_buffer.add( std::string(data,n) ) ;
while( m_buffer.more() )
{
if( ! processLine( m_buffer.line() ) )
return ;
}
}
bool GSmtp::AdminPeer::processLine( const std::string & line )
{
if( is(line,"FLUSH") )
{
flush( m_server_address ) ;
}
else if( is(line,"HELP") )
{
help() ;
prompt() ;
}
else if( is(line,"INFO") )
{
info() ;
prompt() ;
}
else if( is(line,"NOTIFY") )
{
m_notifying = true ;
prompt() ;
}
else if( is(line,"LIST") )
{
list() ;
prompt() ;
}
else if( is(line,"QUIT") )
{
doDelete() ;
return false ;
}
else if( is(line,"TERMINATE") && m_with_terminate )
{
if( GNet::EventLoop::exists() )
GNet::EventLoop::instance().quit() ;
}
else if( line.find_first_not_of(" \r\n\t") != std::string::npos )
{
send( "error: unrecognised command" ) ;
prompt() ;
}
else
{
prompt() ;
}
return true ;
}
//static
std::string GSmtp::AdminPeer::crlf()
{
return "\015\012" ;
}
//static
bool GSmtp::AdminPeer::is( const std::string & line_in , const char * key )
{
std::string line( line_in ) ;
G::Str::trim( line , " \t" ) ;
G::Str::toUpper( line ) ;
return line.find(key) == 0U ;
}
void GSmtp::AdminPeer::help()
{
send( "commands: flush, help, info, list, notify, quit" ) ;
}
void GSmtp::AdminPeer::flush( const std::string & address )
{
G_DEBUG( "GSmtp::AdminPeer: flush: \"" << address << "\"" ) ;
if( m_client.get() != NULL && m_client->busy() )
{
send( "error: still working" ) ;
}
else if( address.empty() )
{
send( "error: no remote server configured: use --forward-to" ) ;
}
else
{
const bool quit_on_disconnect = false ;
m_client <<= new Client( m_server.store() , m_server.secrets() ,
quit_on_disconnect , m_server.responseTimeout() ) ;
m_client->doneSignal().connect( G::slot(*this,&AdminPeer::clientDone) ) ;
std::string rc = m_client->startSending( address , m_server.connectionTimeout() ) ;
if( rc.length() != 0U )
{
send( std::string("error: ") + rc ) ;
}
}
}
void GSmtp::AdminPeer::prompt()
{
std::string p( "E-MailRelay> " ) ;
ssize_t rc = socket().write( p.data() , p.length() ) ;
if( rc < 0 || static_cast<size_t>(rc) < p.length() )
doDelete() ; // onDelete() and "delete this"
}
void GSmtp::AdminPeer::send( std::string line )
{
line.append( crlf() ) ;
ssize_t rc = socket().write( line.data() , line.length() ) ;
if( rc < 0 || static_cast<size_t>(rc) < line.length() )
doDelete() ; // onDelete() and "delete this"
}
void GSmtp::AdminPeer::notify( const std::string & s0 , const std::string & s1 , const std::string & s2 )
{
if( m_notifying )
send( crlf() + "EVENT: " + s0 + ": " + s1 + ": " + s2 ) ;
}
void GSmtp::AdminPeer::info()
{
std::ostringstream ss ;
if( GNet::Monitor::instance() )
{
GNet::Monitor::instance()->report( ss , "" , crlf() ) ;
send( ss.str() ) ;
}
else
{
send( "no info" ) ;
}
}
void GSmtp::AdminPeer::list()
{
std::ostringstream ss ;
MessageStore::Iterator iter( m_server.store().iterator(false) ) ;
for(;;)
{
std::auto_ptr<StoredMessage> message( iter.next() ) ;
if( message.get() == NULL ) break ;
ss << message->name() << crlf() ;
}
std::string result = ss.str() ;
if( result.size() == 0U )
send( "<none>" ) ;
else
send( ss.str() ) ;
}
// ===
GSmtp::AdminServer::AdminServer( MessageStore & store , const Secrets & secrets ,
const GNet::Address & listening_address , bool allow_remote ,
const std::string & address , unsigned int response_timeout ,
unsigned int connection_timeout , bool with_terminate ) :
GNet::Server( listening_address ) ,
m_store( store ) ,
m_secrets( secrets ) ,
m_allow_remote( allow_remote ) ,
m_server_address( address ) ,
m_response_timeout( response_timeout ) ,
m_connection_timeout( connection_timeout ) ,
m_with_terminate( with_terminate )
{
G_DEBUG( "GSmtp::AdminServer: administrative interface listening on " << listening_address.displayString() ) ;
}
GSmtp::AdminServer::~AdminServer()
{
// early cleanup so peers can call unregister() safely
serverCleanup() ; // GNet::Server
}
GNet::ServerPeer * GSmtp::AdminServer::newPeer( GNet::Server::PeerInfo peer_info )
{
AdminPeer * peer = new AdminPeer( peer_info , *this , m_server_address , m_with_terminate ) ;
m_peers.push_back( peer ) ;
return peer ;
}
void GSmtp::AdminServer::report() const
{
// no-op
}
void GSmtp::AdminServer::notify( const std::string & s0 , const std::string & s1 , const std::string & s2 )
{
for( PeerList::iterator p = m_peers.begin() ; p != m_peers.end() ; ++p )
{
G_DEBUG( "GSmtp::AdminServer::notify: " << (*p) << ": " << s0 << ": " << s1 ) ;
(*p)->notify( s0 , s1 , s2 ) ;
}
}
void GSmtp::AdminServer::unregister( AdminPeer * peer )
{
G_DEBUG( "GSmtp::AdminServer::unregister: server=" << this << ": peer=" << peer ) ;
PeerList::iterator p = std::find( m_peers.begin() , m_peers.end() , peer ) ;
if( p != m_peers.end() )
m_peers.erase( p ) ;
}
GSmtp::MessageStore & GSmtp::AdminServer::store()
{
return m_store ;
}
const GSmtp::Secrets & GSmtp::AdminServer::secrets() const
{
return m_secrets ;
}
unsigned int GSmtp::AdminServer::responseTimeout() const
{
return m_response_timeout ;
}
unsigned int GSmtp::AdminServer::connectionTimeout() const
{
return m_connection_timeout ;
}