• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //
2 // server.cpp
3 // ~~~~~~~~~~
4 //
5 // Copyright (c) 2003-2021 Christopher M. Kohlhoff (chris at kohlhoff dot com)
6 //
7 // Distributed under the Boost Software License, Version 1.0. (See accompanying
8 // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
9 //
10 
11 #include <boost/asio.hpp>
12 #include <boost/bind/bind.hpp>
13 #include <boost/lexical_cast.hpp>
14 #include <iostream>
15 #include <vector>
16 #include "connection.hpp" // Must come before boost/serialization headers.
17 #include <boost/serialization/vector.hpp>
18 #include "stock.hpp"
19 
20 namespace s11n_example {
21 
22 /// Serves stock quote information to any client that connects to it.
23 class server
24 {
25 public:
26   /// Constructor opens the acceptor and starts waiting for the first incoming
27   /// connection.
server(boost::asio::io_context & io_context,unsigned short port)28   server(boost::asio::io_context& io_context, unsigned short port)
29     : acceptor_(io_context,
30         boost::asio::ip::tcp::endpoint(boost::asio::ip::tcp::v4(), port))
31   {
32     // Create the data to be sent to each client.
33     stock s;
34     s.code = "ABC";
35     s.name = "A Big Company";
36     s.open_price = 4.56;
37     s.high_price = 5.12;
38     s.low_price = 4.33;
39     s.last_price = 4.98;
40     s.buy_price = 4.96;
41     s.buy_quantity = 1000;
42     s.sell_price = 4.99;
43     s.sell_quantity = 2000;
44     stocks_.push_back(s);
45     s.code = "DEF";
46     s.name = "Developer Entertainment Firm";
47     s.open_price = 20.24;
48     s.high_price = 22.88;
49     s.low_price = 19.50;
50     s.last_price = 19.76;
51     s.buy_price = 19.72;
52     s.buy_quantity = 34000;
53     s.sell_price = 19.85;
54     s.sell_quantity = 45000;
55     stocks_.push_back(s);
56 
57     // Start an accept operation for a new connection.
58     connection_ptr new_conn(new connection(acceptor_.get_executor()));
59     acceptor_.async_accept(new_conn->socket(),
60         boost::bind(&server::handle_accept, this,
61           boost::asio::placeholders::error, new_conn));
62   }
63 
64   /// Handle completion of a accept operation.
handle_accept(const boost::system::error_code & e,connection_ptr conn)65   void handle_accept(const boost::system::error_code& e, connection_ptr conn)
66   {
67     if (!e)
68     {
69       // Successfully accepted a new connection. Send the list of stocks to the
70       // client. The connection::async_write() function will automatically
71       // serialize the data structure for us.
72       conn->async_write(stocks_,
73           boost::bind(&server::handle_write, this,
74             boost::asio::placeholders::error, conn));
75     }
76 
77     // Start an accept operation for a new connection.
78     connection_ptr new_conn(new connection(acceptor_.get_executor()));
79     acceptor_.async_accept(new_conn->socket(),
80         boost::bind(&server::handle_accept, this,
81           boost::asio::placeholders::error, new_conn));
82   }
83 
84   /// Handle completion of a write operation.
handle_write(const boost::system::error_code & e,connection_ptr conn)85   void handle_write(const boost::system::error_code& e, connection_ptr conn)
86   {
87     // Nothing to do. The socket will be closed automatically when the last
88     // reference to the connection object goes away.
89   }
90 
91 private:
92   /// The acceptor object used to accept incoming socket connections.
93   boost::asio::ip::tcp::acceptor acceptor_;
94 
95   /// The data to be sent to each client.
96   std::vector<stock> stocks_;
97 };
98 
99 } // namespace s11n_example
100 
main(int argc,char * argv[])101 int main(int argc, char* argv[])
102 {
103   try
104   {
105     // Check command line arguments.
106     if (argc != 2)
107     {
108       std::cerr << "Usage: server <port>" << std::endl;
109       return 1;
110     }
111     unsigned short port = boost::lexical_cast<unsigned short>(argv[1]);
112 
113     boost::asio::io_context io_context;
114     s11n_example::server server(io_context, port);
115     io_context.run();
116   }
117   catch (std::exception& e)
118   {
119     std::cerr << e.what() << std::endl;
120   }
121 
122   return 0;
123 }
124