1 //
2 // server.cpp
3 // ~~~~~~~~~~
4 //
5 // Copyright (c) 2003-2020 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 <cstdlib>
12 #include <iostream>
13 #include <boost/bind/bind.hpp>
14 #include <boost/asio.hpp>
15 #include <boost/asio/ssl.hpp>
16
17 typedef boost::asio::ssl::stream<boost::asio::ip::tcp::socket> ssl_socket;
18
19 class session
20 {
21 public:
session(boost::asio::io_context & io_context,boost::asio::ssl::context & context)22 session(boost::asio::io_context& io_context,
23 boost::asio::ssl::context& context)
24 : socket_(io_context, context)
25 {
26 }
27
socket()28 ssl_socket::lowest_layer_type& socket()
29 {
30 return socket_.lowest_layer();
31 }
32
start()33 void start()
34 {
35 socket_.async_handshake(boost::asio::ssl::stream_base::server,
36 boost::bind(&session::handle_handshake, this,
37 boost::asio::placeholders::error));
38 }
39
handle_handshake(const boost::system::error_code & error)40 void handle_handshake(const boost::system::error_code& error)
41 {
42 if (!error)
43 {
44 socket_.async_read_some(boost::asio::buffer(data_, max_length),
45 boost::bind(&session::handle_read, this,
46 boost::asio::placeholders::error,
47 boost::asio::placeholders::bytes_transferred));
48 }
49 else
50 {
51 delete this;
52 }
53 }
54
handle_read(const boost::system::error_code & error,size_t bytes_transferred)55 void handle_read(const boost::system::error_code& error,
56 size_t bytes_transferred)
57 {
58 if (!error)
59 {
60 boost::asio::async_write(socket_,
61 boost::asio::buffer(data_, bytes_transferred),
62 boost::bind(&session::handle_write, this,
63 boost::asio::placeholders::error));
64 }
65 else
66 {
67 delete this;
68 }
69 }
70
handle_write(const boost::system::error_code & error)71 void handle_write(const boost::system::error_code& error)
72 {
73 if (!error)
74 {
75 socket_.async_read_some(boost::asio::buffer(data_, max_length),
76 boost::bind(&session::handle_read, this,
77 boost::asio::placeholders::error,
78 boost::asio::placeholders::bytes_transferred));
79 }
80 else
81 {
82 delete this;
83 }
84 }
85
86 private:
87 ssl_socket socket_;
88 enum { max_length = 1024 };
89 char data_[max_length];
90 };
91
92 class server
93 {
94 public:
server(boost::asio::io_context & io_context,unsigned short port)95 server(boost::asio::io_context& io_context, unsigned short port)
96 : io_context_(io_context),
97 acceptor_(io_context,
98 boost::asio::ip::tcp::endpoint(boost::asio::ip::tcp::v4(), port)),
99 context_(boost::asio::ssl::context::sslv23)
100 {
101 context_.set_options(
102 boost::asio::ssl::context::default_workarounds
103 | boost::asio::ssl::context::no_sslv2
104 | boost::asio::ssl::context::single_dh_use);
105 context_.set_password_callback(boost::bind(&server::get_password, this));
106 context_.use_certificate_chain_file("server.pem");
107 context_.use_private_key_file("server.pem", boost::asio::ssl::context::pem);
108 context_.use_tmp_dh_file("dh2048.pem");
109
110 start_accept();
111 }
112
get_password() const113 std::string get_password() const
114 {
115 return "test";
116 }
117
start_accept()118 void start_accept()
119 {
120 session* new_session = new session(io_context_, context_);
121 acceptor_.async_accept(new_session->socket(),
122 boost::bind(&server::handle_accept, this, new_session,
123 boost::asio::placeholders::error));
124 }
125
handle_accept(session * new_session,const boost::system::error_code & error)126 void handle_accept(session* new_session,
127 const boost::system::error_code& error)
128 {
129 if (!error)
130 {
131 new_session->start();
132 }
133 else
134 {
135 delete new_session;
136 }
137
138 start_accept();
139 }
140
141 private:
142 boost::asio::io_context& io_context_;
143 boost::asio::ip::tcp::acceptor acceptor_;
144 boost::asio::ssl::context context_;
145 };
146
main(int argc,char * argv[])147 int main(int argc, char* argv[])
148 {
149 try
150 {
151 if (argc != 2)
152 {
153 std::cerr << "Usage: server <port>\n";
154 return 1;
155 }
156
157 boost::asio::io_context io_context;
158
159 using namespace std; // For atoi.
160 server s(io_context, atoi(argv[1]));
161
162 io_context.run();
163 }
164 catch (std::exception& e)
165 {
166 std::cerr << "Exception: " << e.what() << "\n";
167 }
168
169 return 0;
170 }
171