sample/raw_channel/tcp_srv.cpp

/*
  Пример, демонстрирующий работу с серверным raw-соединением.

  Создается серверное соединение и отображаются события,
  которые происходят с этим соединением.
*/

#include <iostream>

#include <ace/OS_main.h>

// Базовые заголовочные файлы SObjectizer.
#include <so_4/rt/h/rt.hpp>
#include <so_4/api/h/api.hpp>

// Используем простой диспетчер с одной рабочей нитью.
#include <so_4/timer_thread/simple/h/pub.hpp>
#include <so_4/disp/one_thread/h/pub.hpp>

// Описание агента, который обслуживает серверный
// raw-канал и средств для создания каналов.
#include <so_4/rt/comm/h/a_raw_incoming_channel_processor.hpp>
#include <so_4/transport_layer/socket/h/pub.hpp>

//
// Класс агента, который будет работать с серверным соединением.
//
class a_main_t :
  public so_4::rt::agent_t
{
  typedef so_4::rt::agent_t base_type_t;

  public :
    a_main_t();
    virtual ~a_main_t();

    virtual const char *
    so_query_type() const;

    virtual void
    so_on_subscription();

    // Имя главного тестового агента, который должен
    // присутствовать в приложении в единственном числе.
    static std::string &
    agent_name();

    // Имя агента серверного соединения, который должен
    // присутствовать в приложении в единственном числе.
    static std::string &
    tcp_agent_name();

    // Реакция на успешное создание серверного сокета.
    void
    evt_success(
      const so_4::rt::comm::msg_success & );

    // Реакция на неудачное создание серверного сокета.
    void
    evt_fail(
      const so_4::rt::comm::msg_fail & );

    // Реакция на подключение клиента.
    void
    evt_client_connected(
      const so_4::rt::comm::msg_client_connected & cmd );

    // Реакция на отключение клиента.
    void
    evt_client_disconnected(
      const so_4::rt::comm::msg_client_disconnected & cmd );

    // Реакция на поступление данных в канал.
    void
    evt_incoming_data(
      const so_4::rt::comm::msg_raw_package & cmd );

};

SOL4_CLASS_START( a_main_t )

  SOL4_EVENT_STC(
    evt_success,
    so_4::rt::comm::msg_success )
  SOL4_EVENT_STC(
    evt_fail,
    so_4::rt::comm::msg_fail )
  SOL4_EVENT_STC(
    evt_client_connected,
    so_4::rt::comm::msg_client_connected )
  SOL4_EVENT_STC(
    evt_client_disconnected,
    so_4::rt::comm::msg_client_disconnected )
  SOL4_EVENT_STC(
    evt_incoming_data,
    so_4::rt::comm::msg_raw_package )

  SOL4_STATE_START( st_normal )
    SOL4_STATE_EVENT( evt_success )
    SOL4_STATE_EVENT( evt_fail )
    SOL4_STATE_EVENT( evt_client_connected )
    SOL4_STATE_EVENT( evt_client_disconnected )
    SOL4_STATE_EVENT( evt_incoming_data )
  SOL4_STATE_FINISH()

SOL4_CLASS_FINISH()

a_main_t::a_main_t()
:
  base_type_t( agent_name().c_str() )
{
}

a_main_t::~a_main_t()
{
}

void
a_main_t::so_on_subscription()
{
  so_subscribe( "evt_success", tcp_agent_name(), "msg_success" );

  so_subscribe( "evt_fail", tcp_agent_name(), "msg_fail" );

  so_subscribe( "evt_client_connected", tcp_agent_name(),
    "msg_client_connected" );

  so_subscribe( "evt_client_disconnected", tcp_agent_name(),
    "msg_client_disconnected" );

  so_subscribe( "evt_incoming_data", tcp_agent_name(),
    "msg_raw_package" );
}

std::string &
a_main_t::agent_name()
{
  static std::string name( "a_main" );

  return name;
}

std::string &
a_main_t::tcp_agent_name()
{
  static std::string name( "a_tcp_srvsock" );

  return name;
}

void
a_main_t::evt_success(
  const so_4::rt::comm::msg_success & )
{
  std::cout << so_query_name() << ".evt_success" << std::endl;
}

void
a_main_t::evt_fail(
  const so_4::rt::comm::msg_fail & cmd )
{
  std::cout << so_query_name() << ".evt_fail: "
    << cmd.m_reason << std::endl;
}

void
a_main_t::evt_client_connected(
  const so_4::rt::comm::msg_client_connected & cmd )
{
  std::cout << so_query_name() << ".evt_client_connected: "
    << cmd.m_channel << std::endl;
}

void
a_main_t::evt_client_disconnected(
  const so_4::rt::comm::msg_client_disconnected & cmd )
{
  std::cout << so_query_name() << ".evt_client_disconnected: "
    << cmd.m_channel << std::endl;
}

void
a_main_t::evt_incoming_data(
  const so_4::rt::comm::msg_raw_package & cmd )
{
  std::cout << so_query_name() << ".evt_incoming_data: "
    << cmd.m_channel
    << "\n\tdata size: " << cmd.m_package.size() << std::endl;

  std::string v(
    (const char *)cmd.m_package.ptr(),
    cmd.m_package.size() );
  std::cout << v << std::endl;

  // Если канал оказался заблокированным, то разблокируем его.
  cmd.unblock_channel();
}

// Создание главной кооперации примера.
so_4::rt::agent_coop_t *
create_coop( const char * ip_address )
{
  a_main_t * a_main = new a_main_t();

  so_4::rt::comm::a_raw_incoming_channel_processor_t * a_tcp_srvsock =
    new so_4::rt::comm::a_raw_incoming_channel_processor_t(
      a_main_t::tcp_agent_name(),
      so_4::transport_layer::socket::create_acceptor_controller(
          ip_address ) );

  so_4::rt::agent_t * agents[] = {
    a_main, a_tcp_srvsock
  };

  return new so_4::rt::dyn_agent_coop_t( "srvsock1",
      agents, sizeof( agents ) / sizeof( agents[ 0 ] ) );
}

int
main( int argc, char ** argv )
{
  if( 2 == argc ) {
    // Создаем таймер, диспетчер. Затем запускаем
    // run-time SObjectizer-а.
    so_4::ret_code_t rc = so_4::api::start(
      // Диспетчер будет уничтожен при выходе из start().
      so_4::disp::one_thread::create_disp(
        // Таймер будет уничтожен диспетчером.
        so_4::timer_thread::simple::create_timer_thread(),
        so_4::auto_destroy_timer ),
      so_4::auto_destroy_disp,
      create_coop( argv[ 1 ] ) );
    if( rc )
    {
      // Ошибка старта.
      std::cerr << "start: " << rc << std::endl;
    }

    return rc;
  }
  else
    std::cerr << "sample_raw_channel_tcp_srv <ip-address>" << std::endl;

  return 0;
}

Документация по SObjectizer v.4.4 'Тебуломста'. Последние изменения: Thu Sep 18 10:26:46 2008. Создано системой  doxygen1.5.6 Intervale SourceForge.net Logo