Wire Sysio Wire Sysion 1.0.0
Loading...
Searching...
No Matches
utility_client.cpp
Go to the documentation of this file.
1/*
2 * Copyright (c) 2014, Peter Thorson. All rights reserved.
3 *
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions are met:
6 * * Redistributions of source code must retain the above copyright
7 * notice, this list of conditions and the following disclaimer.
8 * * Redistributions in binary form must reproduce the above copyright
9 * notice, this list of conditions and the following disclaimer in the
10 * documentation and/or other materials provided with the distribution.
11 * * Neither the name of the WebSocket++ Project nor the
12 * names of its contributors may be used to endorse or promote products
13 * derived from this software without specific prior written permission.
14 *
15 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
16 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
17 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
18 * ARE DISCLAIMED. IN NO EVENT SHALL PETER THORSON BE LIABLE FOR ANY
19 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
20 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
21 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
22 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
23 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
24 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
25 */
26
27// **NOTE:** This file is a snapshot of the WebSocket++ utility client tutorial.
28// Additional related material can be found in the tutorials/utility_client
29// directory of the WebSocket++ repository.
30
33
36
37#include <cstdlib>
38#include <iostream>
39#include <map>
40#include <string>
41#include <sstream>
42
44
46public:
47 typedef websocketpp::lib::shared_ptr<connection_metadata> ptr;
48
50 : m_id(id)
51 , m_hdl(hdl)
52 , m_status("Connecting")
53 , m_uri(uri)
54 , m_server("N/A")
55 {}
56
58 m_status = "Open";
59
61 m_server = con->get_response_header("Server");
62 }
63
65 m_status = "Failed";
66
68 m_server = con->get_response_header("Server");
69 m_error_reason = con->get_ec().message();
70 }
71
73 m_status = "Closed";
75 std::stringstream s;
76 s << "close code: " << con->get_remote_close_code() << " ("
77 << websocketpp::close::status::get_string(con->get_remote_close_code())
78 << "), close reason: " << con->get_remote_close_reason();
79 m_error_reason = s.str();
80 }
81
83 if (msg->get_opcode() == websocketpp::frame::opcode::text) {
84 m_messages.push_back("<< " + msg->get_payload());
85 } else {
86 m_messages.push_back("<< " + websocketpp::utility::to_hex(msg->get_payload()));
87 }
88 }
89
91 return m_hdl;
92 }
93
94 int get_id() const {
95 return m_id;
96 }
97
98 std::string get_status() const {
99 return m_status;
100 }
101
102 void record_sent_message(std::string message) {
103 m_messages.push_back(">> " + message);
104 }
105
106 friend std::ostream & operator<< (std::ostream & out, connection_metadata const & data);
107private:
108 int m_id;
110 std::string m_status;
111 std::string m_uri;
112 std::string m_server;
113 std::string m_error_reason;
114 std::vector<std::string> m_messages;
115};
116
117std::ostream & operator<< (std::ostream & out, connection_metadata const & data) {
118 out << "> URI: " << data.m_uri << "\n"
119 << "> Status: " << data.m_status << "\n"
120 << "> Remote Server: " << (data.m_server.empty() ? "None Specified" : data.m_server) << "\n"
121 << "> Error/close reason: " << (data.m_error_reason.empty() ? "N/A" : data.m_error_reason) << "\n";
122 out << "> Messages Processed: (" << data.m_messages.size() << ") \n";
123
124 std::vector<std::string>::const_iterator it;
125 for (it = data.m_messages.begin(); it != data.m_messages.end(); ++it) {
126 out << *it << "\n";
127 }
128
129 return out;
130}
131
132class websocket_endpoint {
133public:
134 websocket_endpoint () : m_next_id(0) {
137
138 m_endpoint.init_asio();
139 m_endpoint.start_perpetual();
140
141 m_thread = websocketpp::lib::make_shared<websocketpp::lib::thread>(&client::run, &m_endpoint);
142 }
143
145 m_endpoint.stop_perpetual();
146
147 for (con_list::const_iterator it = m_connection_list.begin(); it != m_connection_list.end(); ++it) {
148 if (it->second->get_status() != "Open") {
149 // Only close open connections
150 continue;
151 }
152
153 std::cout << "> Closing connection " << it->second->get_id() << std::endl;
154
155 websocketpp::lib::error_code ec;
156 m_endpoint.close(it->second->get_hdl(), websocketpp::close::status::going_away, "", ec);
157 if (ec) {
158 std::cout << "> Error closing connection " << it->second->get_id() << ": "
159 << ec.message() << std::endl;
160 }
161 }
162
163 m_thread->join();
164 }
165
166 int connect(std::string const & uri) {
167 websocketpp::lib::error_code ec;
168
169 client::connection_ptr con = m_endpoint.get_connection(uri, ec);
170
171 if (ec) {
172 std::cout << "> Connect initialization error: " << ec.message() << std::endl;
173 return -1;
174 }
175
176 int new_id = m_next_id++;
177 connection_metadata::ptr metadata_ptr = websocketpp::lib::make_shared<connection_metadata>(new_id, con->get_handle(), uri);
178 m_connection_list[new_id] = metadata_ptr;
179
180 con->set_open_handler(websocketpp::lib::bind(
182 metadata_ptr,
183 &m_endpoint,
184 websocketpp::lib::placeholders::_1
185 ));
186 con->set_fail_handler(websocketpp::lib::bind(
188 metadata_ptr,
189 &m_endpoint,
190 websocketpp::lib::placeholders::_1
191 ));
192 con->set_close_handler(websocketpp::lib::bind(
194 metadata_ptr,
195 &m_endpoint,
196 websocketpp::lib::placeholders::_1
197 ));
198 con->set_message_handler(websocketpp::lib::bind(
200 metadata_ptr,
201 websocketpp::lib::placeholders::_1,
202 websocketpp::lib::placeholders::_2
203 ));
204
205 m_endpoint.connect(con);
206
207 return new_id;
208 }
209
210 void close(int id, websocketpp::close::status::value code, std::string reason) {
211 websocketpp::lib::error_code ec;
212
213 con_list::iterator metadata_it = m_connection_list.find(id);
214 if (metadata_it == m_connection_list.end()) {
215 std::cout << "> No connection found with id " << id << std::endl;
216 return;
217 }
218
219 m_endpoint.close(metadata_it->second->get_hdl(), code, reason, ec);
220 if (ec) {
221 std::cout << "> Error initiating close: " << ec.message() << std::endl;
222 }
223 }
224
225 void send(int id, std::string message) {
226 websocketpp::lib::error_code ec;
227
228 con_list::iterator metadata_it = m_connection_list.find(id);
229 if (metadata_it == m_connection_list.end()) {
230 std::cout << "> No connection found with id " << id << std::endl;
231 return;
232 }
233
234 m_endpoint.send(metadata_it->second->get_hdl(), message, websocketpp::frame::opcode::text, ec);
235 if (ec) {
236 std::cout << "> Error sending message: " << ec.message() << std::endl;
237 return;
238 }
239
240 metadata_it->second->record_sent_message(message);
241 }
242
244 con_list::const_iterator metadata_it = m_connection_list.find(id);
245 if (metadata_it == m_connection_list.end()) {
247 } else {
248 return metadata_it->second;
249 }
250 }
251private:
252 typedef std::map<int,connection_metadata::ptr> con_list;
253
254 client m_endpoint;
255 websocketpp::lib::shared_ptr<websocketpp::lib::thread> m_thread;
256
257 con_list m_connection_list;
258 int m_next_id;
259};
260
261int main() {
262 bool done = false;
263 std::string input;
264 websocket_endpoint endpoint;
265
266 while (!done) {
267 std::cout << "Enter Command: ";
268 std::getline(std::cin, input);
269
270 if (input == "quit") {
271 done = true;
272 } else if (input == "help") {
273 std::cout
274 << "\nCommand List:\n"
275 << "connect <ws uri>\n"
276 << "send <connection id> <message>\n"
277 << "close <connection id> [<close code:default=1000>] [<close reason>]\n"
278 << "show <connection id>\n"
279 << "help: Display this help text\n"
280 << "quit: Exit the program\n"
281 << std::endl;
282 } else if (input.substr(0,7) == "connect") {
283 int id = endpoint.connect(input.substr(8));
284 if (id != -1) {
285 std::cout << "> Created connection with id " << id << std::endl;
286 }
287 } else if (input.substr(0,4) == "send") {
288 std::stringstream ss(input);
289
290 std::string cmd;
291 int id;
292 std::string message;
293
294 ss >> cmd >> id;
295 std::getline(ss,message);
296
297 endpoint.send(id, message);
298 } else if (input.substr(0,5) == "close") {
299 std::stringstream ss(input);
300
301 std::string cmd;
302 int id;
303 int close_code = websocketpp::close::status::normal;
304 std::string reason;
305
306 ss >> cmd >> id >> close_code;
307 std::getline(ss,reason);
308
309 endpoint.close(id, close_code, reason);
310 } else if (input.substr(0,4) == "show") {
311 int id = atoi(input.substr(5).c_str());
312
313 connection_metadata::ptr metadata = endpoint.get_metadata(id);
314 if (metadata) {
315 std::cout << *metadata << std::endl;
316 } else {
317 std::cout << "> Unknown connection id " << id << std::endl;
318 }
319 } else {
320 std::cout << "> Unrecognized Command" << std::endl;
321 }
322 }
323
324 return 0;
325}
void on_fail(client *c, websocketpp::connection_hdl hdl)
friend std::ostream & operator<<(std::ostream &out, connection_metadata const &data)
websocketpp::connection_hdl get_hdl() const
void on_message(websocketpp::connection_hdl, client::message_ptr msg)
void record_sent_message(std::string message)
void on_open(client *c, websocketpp::connection_hdl hdl)
connection_metadata(int id, websocketpp::connection_hdl hdl, std::string uri)
void on_close(client *c, websocketpp::connection_hdl hdl)
websocketpp::lib::shared_ptr< connection_metadata > ptr
std::string get_status() const
int connect(std::string const &uri)
connection_metadata::ptr get_metadata(int id) const
void send(int id, std::string message)
void close(int id, websocketpp::close::status::value code, std::string reason)
Client endpoint role based on the given config.
connection_ptr connect(connection_ptr con)
Begin the connection process for the given connection.
connection_ptr get_connection(uri_ptr location, lib::error_code &ec)
Get a new connection.
connection_ptr get_con_from_hdl(connection_hdl hdl, lib::error_code &ec)
Retrieves a connection_ptr from a connection_hdl (exception free)
Definition endpoint.hpp:643
void clear_access_channels(log::level channels)
Clear Access logging channels.
Definition endpoint.hpp:231
void send(connection_hdl hdl, std::string const &payload, frame::opcode::value op, lib::error_code &ec)
Create a message and add it to the outgoing send queue (exception free)
void clear_error_channels(log::level channels)
Clear Error logging channels.
Definition endpoint.hpp:253
void close(connection_hdl hdl, close::status::value const code, std::string const &reason, lib::error_code &ec)
uint64_t id
Definition code_cache.cpp:0
std::string get_string(value code)
Return a human readable interpretation of a WebSocket close code.
Definition close.hpp:227
uint16_t value
The type of a close code value.
Definition close.hpp:49
std::string to_hex(std::string const &input)
Convert std::string to ascii printed string of hex digits.
lib::weak_ptr< void > connection_hdl
A handle to uniquely identify a connection.
static level const all
Special aggregate value representing "all levels".
Definition levels.hpp:152
static level const all
Special aggregate value representing "all levels".
Definition levels.hpp:80
std::ostream & operator<<(std::ostream &out, connection_metadata const &data)
int main()
websocketpp::client< websocketpp::config::asio_client > client
char * s