224 lines
5.8 KiB
C++
224 lines
5.8 KiB
C++
//
|
|
// Copyright (C) 2001-2018 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/>.
|
|
// ===
|
|
//
|
|
// emailrelay_test_verifier.cpp
|
|
//
|
|
// A dummy network address verifier for testing "emailrelay --verifier net:<transport-address>".
|
|
//
|
|
// usage: emailrelay-test-verifier [--ipv6] [--port <port>] [--log] [--log-file <file>] [--debug] [--pid-file <pidfile>]
|
|
//
|
|
// The action of the verifier is dictated by special sub-strings in the
|
|
// recipient:
|
|
// * OK -- valid
|
|
// * L -- local
|
|
// * X -- no response (to test response timeouts)
|
|
// * x -- disconnect
|
|
// * ! -- abort
|
|
//
|
|
// Listens on port 10020 by default.
|
|
//
|
|
|
|
#include "gdef.h"
|
|
#include "gserver.h"
|
|
#include "glinebuffer.h"
|
|
#include "gstr.h"
|
|
#include "gevent.h"
|
|
#include "gprocess.h"
|
|
#include "gbufferedserverpeer.h"
|
|
#include "ggetopt.h"
|
|
#include "garg.h"
|
|
#include "gsleep.h"
|
|
#include "gdebug.h"
|
|
#include <sstream>
|
|
#include <iostream>
|
|
#include <fstream>
|
|
|
|
namespace Main
|
|
{
|
|
class VerifierPeer ;
|
|
class Verifier ;
|
|
}
|
|
|
|
class Main::VerifierPeer : public GNet::BufferedServerPeer
|
|
{
|
|
public:
|
|
explicit VerifierPeer( GNet::Server::PeerInfo info ) ;
|
|
private:
|
|
virtual void onDelete( const std::string & ) ;
|
|
virtual bool onReceive( const char * , size_t , size_t ) ;
|
|
virtual void onSecure( const std::string & ) ;
|
|
virtual void onSendComplete() {}
|
|
bool processLine( std::string ) ;
|
|
} ;
|
|
|
|
Main::VerifierPeer::VerifierPeer( GNet::Server::PeerInfo info ) :
|
|
BufferedServerPeer( info , GNet::LineBufferConfig::newline() )
|
|
{
|
|
G_LOG_S( "VerifierPeer::ctor: new connection from " << info.m_address.displayString() ) ;
|
|
}
|
|
|
|
void Main::VerifierPeer::onDelete( const std::string & )
|
|
{
|
|
G_LOG_S( "VerifierPeer::onDelete: disconnected" ) ;
|
|
}
|
|
|
|
void Main::VerifierPeer::onSecure( const std::string & )
|
|
{
|
|
}
|
|
|
|
bool Main::VerifierPeer::onReceive( const char * line_data , size_t line_size , size_t )
|
|
{
|
|
std::string line( line_data , line_size ) ;
|
|
G_DEBUG( "VerifierPeer::onReceive: [" << G::Str::printable(line) << "]" ) ;
|
|
processLine( line ) ;
|
|
return true ;
|
|
}
|
|
|
|
bool Main::VerifierPeer::processLine( std::string line )
|
|
{
|
|
G_LOG_S( "VerifierPeer::processLine: line: \"" << line << "\"" ) ;
|
|
|
|
G::StringArray part ;
|
|
G::Str::splitIntoFields( line , part , "|" ) ;
|
|
std::string to = part.at(0U) ;
|
|
|
|
bool local = to.find("L") != std::string::npos ;
|
|
bool valid = to.find("OK") != std::string::npos ;
|
|
bool blackhole = to.find("X") != std::string::npos ;
|
|
bool disconnect = to.find("x") != std::string::npos ;
|
|
bool abort = to.find("!") != std::string::npos ;
|
|
|
|
if( abort )
|
|
{
|
|
G_LOG_S( "VerifierPeer::processLine: sending 100" ) ;
|
|
send( "100\n" ) ;
|
|
}
|
|
else if( blackhole )
|
|
{
|
|
G_LOG_S( "VerifierPeer::processLine: sending nothing" ) ;
|
|
}
|
|
else if( disconnect )
|
|
{
|
|
throw std::runtime_error( "disconnection" ) ;
|
|
}
|
|
else if( valid && local )
|
|
{
|
|
G_LOG_S( "VerifierPeer::processLine: sending postmaster" ) ;
|
|
send( "0|postmaster|Postmaster <postmaster@localhost>\n" ) ;
|
|
}
|
|
else if( valid )
|
|
{
|
|
G_LOG_S( "VerifierPeer::processLine: sending valid" ) ;
|
|
send( "1|" + to + "\n" ) ;
|
|
}
|
|
else
|
|
{
|
|
G_LOG_S( "VerifierPeer::processLine: sending error" ) ;
|
|
send( "2|VerifierError\n" ) ;
|
|
}
|
|
return true ;
|
|
}
|
|
|
|
// ===
|
|
|
|
class Main::Verifier : public GNet::Server
|
|
{
|
|
public:
|
|
Verifier( GNet::ExceptionHandler & , bool ipv6 , unsigned int port ) ;
|
|
virtual GNet::ServerPeer * newPeer( GNet::Server::PeerInfo ) override ;
|
|
} ;
|
|
|
|
Main::Verifier::Verifier( GNet::ExceptionHandler & eh , bool ipv6 , unsigned int port ) :
|
|
GNet::Server( eh , GNet::Address(ipv6?GNet::Address::Family::ipv6():GNet::Address::Family::ipv4(),port) )
|
|
{
|
|
}
|
|
|
|
GNet::ServerPeer * Main::Verifier::newPeer( GNet::Server::PeerInfo info )
|
|
{
|
|
return new VerifierPeer( info ) ;
|
|
}
|
|
|
|
// ===
|
|
|
|
static int run( bool ipv6 , unsigned int port )
|
|
{
|
|
struct Eh : GNet::ExceptionHandler
|
|
{
|
|
virtual void onException( std::exception & ) { throw ; }
|
|
} eh ;
|
|
unique_ptr<GNet::EventLoop> loop( GNet::EventLoop::create() ) ;
|
|
GNet::TimerList timer_list ;
|
|
Main::Verifier verifier( eh , ipv6 , port ) ;
|
|
loop->run() ;
|
|
return 0 ;
|
|
}
|
|
|
|
int main( int argc , char * argv [] )
|
|
{
|
|
try
|
|
{
|
|
G::Arg arg( argc , argv ) ;
|
|
G::GetOpt opt( arg ,
|
|
"h!help!show help!!0!!1" "|"
|
|
"l!log!logging!!0!!1" "|"
|
|
"f!log-file!log file!!1!file!1" "|"
|
|
"d!debug!more logging!!0!!1" "|"
|
|
"6!ipv6!use ipv6!!0!!1" "|"
|
|
"P!port!port number!!1!port!1" "|"
|
|
"f!pid-file!pid file!!1!path!1" "|"
|
|
) ;
|
|
if( opt.hasErrors() )
|
|
{
|
|
opt.showErrors(std::cerr) ;
|
|
return 2 ;
|
|
}
|
|
if( opt.contains("help") )
|
|
{
|
|
opt.options().showUsage( std::cout , arg.prefix() ) ;
|
|
return 0 ;
|
|
}
|
|
bool log = opt.contains("log") ;
|
|
bool debug = opt.contains("debug") ;
|
|
std::string log_file = opt.value("log-file","") ;
|
|
bool ipv6 = opt.contains("ipv6") ;
|
|
unsigned int port = G::Str::toUInt( opt.value("port","10020") ) ;
|
|
std::string pid_file = opt.value("pid-file","") ;
|
|
|
|
if( !pid_file.empty() )
|
|
{
|
|
std::ofstream file( pid_file.c_str() ) ;
|
|
file << G::Process::Id().str() << std::endl ;
|
|
}
|
|
|
|
G::LogOutput log_output( log , debug , log_file ) ;
|
|
int rc = run( ipv6 , port ) ;
|
|
std::cout << "done" << std::endl ;
|
|
return rc ;
|
|
}
|
|
catch( std::exception & e )
|
|
{
|
|
std::cerr << "exception: " << e.what() << std::endl ;
|
|
}
|
|
catch(...)
|
|
{
|
|
std::cerr << "exception\n" ;
|
|
}
|
|
return 1 ;
|
|
}
|
|
|
|
/// \file emailrelay_test_verifier.cpp
|