#include <so_5/all.hpp>
#include <iostream>
#include <cstdio>
#include <cstring>
#include <cstdlib>
namespace sample
{
struct cfg_t
{
unsigned int m_request_count{ 1000u };
bool m_separate_dispatchers{ false };
};
[[nodiscard]]
cfg_t
try_parse_cmdline(
int argc,
char ** argv )
{
if( 1 == argc )
{
std::cout << "usage:\n"
"sample.so_5_extra.msg_hierarchy.ping_pong <options>\n"
"\noptions:\n"
"-s, --separate-dispatchers agents should work on "
"different dispatchers\n"
"-r, --requests count of requests to send\n"
<< std::endl;
throw std::runtime_error( "No command-line arguments" );
}
auto is_arg = []( const char * value,
const char * v1,
const char * v2 )
{
return 0 == std::strcmp( value, v1 ) ||
0 == std::strcmp( value, v2 );
};
cfg_t result;
char ** current = argv + 1;
char ** last = argv + argc;
while( current != last )
{
if( is_arg( *current, "-s", "--separate-dispatchers" ) )
{
result.m_separate_dispatchers = true;
}
else if( is_arg( *current, "-r", "--requests" ) )
{
++current;
if( current == last )
throw std::runtime_error( "-r requires argument" );
result.m_request_count = static_cast< unsigned int >(
std::atoi( *current ) );
}
else
{
throw std::runtime_error(
std::string( "unknown argument: " ) + *current );
}
++current;
}
return result;
}
void
show_cfg(
const cfg_t & cfg )
{
std::cout << "Configuration: "
<< "separate dispatchers: "
<< ( cfg.m_separate_dispatchers ? "yes" : "no" )
<< ", requests: " << cfg.m_request_count
<< std::endl;
}
{
basic() = default;
};
struct abstract_ping
: public basic
{
[[nodiscard]]
virtual int
payload() const noexcept = 0;
abstract_ping()
{}
};
struct abstract_pong
: public basic
{
[[nodiscard]]
virtual int
payload() const noexcept = 0;
abstract_pong()
{}
};
template< typename Actual_Ping_Type >
class pinger_t final : public so_5::agent_t
{
const so_5::mbox_t m_out_mbox;
unsigned int m_pings_left;
public :
pinger_t(
context_t ctx,
unsigned int pings_left )
: so_5::agent_t{ std::move(ctx) }
, m_pings_left{ pings_left }
{}
void
so_define_agent() override
{
so_subscribe( m_consumer.receiving_mbox< abstract_pong >() ).event(
[this]( mhood_t<abstract_pong> cmd ) {
if( m_pings_left ) --m_pings_left;
if( m_pings_left )
so_5::send< Actual_Ping_Type >( m_out_mbox, cmd->payload() + 1 );
else
so_environment().stop();
} );
}
void so_evt_start() override
{
so_5::send< Actual_Ping_Type >( m_out_mbox, 0 );
}
};
template< typename Actual_Pong_Type >
class ponger_t final : public so_5::agent_t
{
const so_5::mbox_t m_out_mbox;
public :
ponger_t(
context_t ctx,
: so_5::agent_t{ std::move(ctx) }
{}
void
so_define_agent() override
{
so_subscribe( m_consumer.receiving_mbox< abstract_ping >() ).event(
[this]( mhood_t<abstract_ping> cmd ) {
so_5::send< Actual_Pong_Type >( m_out_mbox, cmd->payload() + 1 );
} );
}
};
struct ping final
: public abstract_ping
{
int m_payload;
int
payload() const noexcept override { return m_payload; }
ping( int payload )
, m_payload{ payload }
{}
};
struct pong final
: public abstract_pong
{
int m_payload;
int
payload() const noexcept override { return m_payload; }
pong( int payload )
, m_payload{ payload }
{}
};
void
run_sample(
const cfg_t & cfg )
{
so_5::launch(
[&cfg]( so_5::environment_t & env )
{
auto first_binder = so_5::disp::one_thread::make_dispatcher(
env, "first" ).binder();
auto second_binder = cfg.m_separate_dispatchers
? so_5::disp::one_thread::make_dispatcher( env, "second" ).binder()
: first_binder;
env.introduce_coop(
[&]( so_5::coop_t & coop )
{
coop.environment(),
};
coop.make_agent_with_binder< pinger_t< ping > >(
first_binder,
demuxer, cfg.m_request_count );
coop.make_agent_with_binder< ponger_t< pong > >(
second_binder,
demuxer );
} );
} );
}
}
using namespace sample;
int main( int argc, char ** argv )
{
try
{
cfg_t cfg = try_parse_cmdline( argc, argv );
show_cfg( cfg );
run_sample( cfg );
return 0;
}
catch( const std::exception & x )
{
std::cerr << "*** Exception caught: " << x.what() << std::endl;
}
return 2;
}
Support for hierarchy of message types.