1 //
2 // Copyright (c) 2016-2019 Vinnie Falco (vinnie dot falco at gmail dot com)
3 //
4 // Distributed under the Boost Software License, Version 1.0. (See accompanying
5 // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
6 //
7 // Official repository: https://github.com/boostorg/beast
8 //
9
10 //------------------------------------------------------------------------------
11 //
12 // Example: WebSocket server, asynchronous
13 //
14 //------------------------------------------------------------------------------
15
16 #include <boost/beast/core.hpp>
17 #include <boost/beast/websocket.hpp>
18 #include <boost/asio/dispatch.hpp>
19 #include <boost/asio/strand.hpp>
20 #include <algorithm>
21 #include <cstdlib>
22 #include <functional>
23 #include <iostream>
24 #include <memory>
25 #include <string>
26 #include <thread>
27 #include <vector>
28
29 namespace beast = boost::beast; // from <boost/beast.hpp>
30 namespace http = beast::http; // from <boost/beast/http.hpp>
31 namespace websocket = beast::websocket; // from <boost/beast/websocket.hpp>
32 namespace net = boost::asio; // from <boost/asio.hpp>
33 using tcp = boost::asio::ip::tcp; // from <boost/asio/ip/tcp.hpp>
34
35 //------------------------------------------------------------------------------
36
37 // Report a failure
38 void
fail(beast::error_code ec,char const * what)39 fail(beast::error_code ec, char const* what)
40 {
41 std::cerr << what << ": " << ec.message() << "\n";
42 }
43
44 // Echoes back all received WebSocket messages
45 class session : public std::enable_shared_from_this<session>
46 {
47 websocket::stream<beast::tcp_stream> ws_;
48 beast::flat_buffer buffer_;
49
50 public:
51 // Take ownership of the socket
52 explicit
session(tcp::socket && socket)53 session(tcp::socket&& socket)
54 : ws_(std::move(socket))
55 {
56 }
57
58 // Get on the correct executor
59 void
run()60 run()
61 {
62 // We need to be executing within a strand to perform async operations
63 // on the I/O objects in this session. Although not strictly necessary
64 // for single-threaded contexts, this example code is written to be
65 // thread-safe by default.
66 net::dispatch(ws_.get_executor(),
67 beast::bind_front_handler(
68 &session::on_run,
69 shared_from_this()));
70 }
71
72 // Start the asynchronous operation
73 void
on_run()74 on_run()
75 {
76 // Set suggested timeout settings for the websocket
77 ws_.set_option(
78 websocket::stream_base::timeout::suggested(
79 beast::role_type::server));
80
81 // Set a decorator to change the Server of the handshake
82 ws_.set_option(websocket::stream_base::decorator(
83 [](websocket::response_type& res)
84 {
85 res.set(http::field::server,
86 std::string(BOOST_BEAST_VERSION_STRING) +
87 " websocket-server-async");
88 }));
89 // Accept the websocket handshake
90 ws_.async_accept(
91 beast::bind_front_handler(
92 &session::on_accept,
93 shared_from_this()));
94 }
95
96 void
on_accept(beast::error_code ec)97 on_accept(beast::error_code ec)
98 {
99 if(ec)
100 return fail(ec, "accept");
101
102 // Read a message
103 do_read();
104 }
105
106 void
do_read()107 do_read()
108 {
109 // Read a message into our buffer
110 ws_.async_read(
111 buffer_,
112 beast::bind_front_handler(
113 &session::on_read,
114 shared_from_this()));
115 }
116
117 void
on_read(beast::error_code ec,std::size_t bytes_transferred)118 on_read(
119 beast::error_code ec,
120 std::size_t bytes_transferred)
121 {
122 boost::ignore_unused(bytes_transferred);
123
124 // This indicates that the session was closed
125 if(ec == websocket::error::closed)
126 return;
127
128 if(ec)
129 fail(ec, "read");
130
131 // Echo the message
132 ws_.text(ws_.got_text());
133 ws_.async_write(
134 buffer_.data(),
135 beast::bind_front_handler(
136 &session::on_write,
137 shared_from_this()));
138 }
139
140 void
on_write(beast::error_code ec,std::size_t bytes_transferred)141 on_write(
142 beast::error_code ec,
143 std::size_t bytes_transferred)
144 {
145 boost::ignore_unused(bytes_transferred);
146
147 if(ec)
148 return fail(ec, "write");
149
150 // Clear the buffer
151 buffer_.consume(buffer_.size());
152
153 // Do another read
154 do_read();
155 }
156 };
157
158 //------------------------------------------------------------------------------
159
160 // Accepts incoming connections and launches the sessions
161 class listener : public std::enable_shared_from_this<listener>
162 {
163 net::io_context& ioc_;
164 tcp::acceptor acceptor_;
165
166 public:
listener(net::io_context & ioc,tcp::endpoint endpoint)167 listener(
168 net::io_context& ioc,
169 tcp::endpoint endpoint)
170 : ioc_(ioc)
171 , acceptor_(ioc)
172 {
173 beast::error_code ec;
174
175 // Open the acceptor
176 acceptor_.open(endpoint.protocol(), ec);
177 if(ec)
178 {
179 fail(ec, "open");
180 return;
181 }
182
183 // Allow address reuse
184 acceptor_.set_option(net::socket_base::reuse_address(true), ec);
185 if(ec)
186 {
187 fail(ec, "set_option");
188 return;
189 }
190
191 // Bind to the server address
192 acceptor_.bind(endpoint, ec);
193 if(ec)
194 {
195 fail(ec, "bind");
196 return;
197 }
198
199 // Start listening for connections
200 acceptor_.listen(
201 net::socket_base::max_listen_connections, ec);
202 if(ec)
203 {
204 fail(ec, "listen");
205 return;
206 }
207 }
208
209 // Start accepting incoming connections
210 void
run()211 run()
212 {
213 do_accept();
214 }
215
216 private:
217 void
do_accept()218 do_accept()
219 {
220 // The new connection gets its own strand
221 acceptor_.async_accept(
222 net::make_strand(ioc_),
223 beast::bind_front_handler(
224 &listener::on_accept,
225 shared_from_this()));
226 }
227
228 void
on_accept(beast::error_code ec,tcp::socket socket)229 on_accept(beast::error_code ec, tcp::socket socket)
230 {
231 if(ec)
232 {
233 fail(ec, "accept");
234 }
235 else
236 {
237 // Create the session and run it
238 std::make_shared<session>(std::move(socket))->run();
239 }
240
241 // Accept another connection
242 do_accept();
243 }
244 };
245
246 //------------------------------------------------------------------------------
247
main(int argc,char * argv[])248 int main(int argc, char* argv[])
249 {
250 // Check command line arguments.
251 if (argc != 4)
252 {
253 std::cerr <<
254 "Usage: websocket-server-async <address> <port> <threads>\n" <<
255 "Example:\n" <<
256 " websocket-server-async 0.0.0.0 8080 1\n";
257 return EXIT_FAILURE;
258 }
259 auto const address = net::ip::make_address(argv[1]);
260 auto const port = static_cast<unsigned short>(std::atoi(argv[2]));
261 auto const threads = std::max<int>(1, std::atoi(argv[3]));
262
263 // The io_context is required for all I/O
264 net::io_context ioc{threads};
265
266 // Create and launch a listening port
267 std::make_shared<listener>(ioc, tcp::endpoint{address, port})->run();
268
269 // Run the I/O service on the requested number of threads
270 std::vector<std::thread> v;
271 v.reserve(threads - 1);
272 for(auto i = threads - 1; i > 0; --i)
273 v.emplace_back(
274 [&ioc]
275 {
276 ioc.run();
277 });
278 ioc.run();
279
280 return EXIT_SUCCESS;
281 }
282