IXWebSocket/ws/ws_chat.cpp

192 lines
5.2 KiB
C++
Raw Normal View History

2018-09-27 23:56:48 +02:00
/*
* ws_chat.cpp
2018-09-27 23:56:48 +02:00
* Author: Benjamin Sergeant
* Copyright (c) 2017-2019 Machine Zone, Inc. All rights reserved.
2018-09-27 23:56:48 +02:00
*/
//
Feature/http (#16) * add skeleton and broken http client code. GET returns "Resource temporarily unavailable" errors... * linux compile fix * can GET some pages * Update formatting in README.md * unittest for sending large messages * document bug * Feature/send large message (#14) * introduce send fragment * can pass a fin frame * can send messages which are a perfect multiple of the chunk size * set fin only for last fragment * cleanup * last fragment should be of type CONTINUATION * Add simple send and receive programs * speedups receiving + better way to wait for thing * receive speedup by using linked list of chunks instead of large array * document bug * use chunks to receive data * trailing spaces * Update README.md Add note about message fragmentation. * Feature/ws cli (#15) * New command line tool for transfering files / still very beta. * add readme * use cli11 for argument parsing * json -> msgpack * stop using base64 and use binary which can be stored in message pack * add target for building with homebrew * all CMakeLists are referenced by the top level one * add ws_chat and ws_connect sub commands to ws * cleanup * add echo and broadcast server as ws sub-commands * add gitignore * comments * ping pong added to ws * mv cobra_publisher under ws folder * Update README.md * linux build fix * linux build fix * move http_client to a ws sub-command * simple HTTP post support (urlencode parameters) * can specify extra headers * chunk encoding / simple redirect support / -I option * follow redirects is optional * make README vim markdown plugin friendly * cleanup argument parsing + add socket creation factory * add missing file * http gzip compression * cleanup * doc * Feature/send large message (#14) * introduce send fragment * can pass a fin frame * can send messages which are a perfect multiple of the chunk size * set fin only for last fragment * cleanup * last fragment should be of type CONTINUATION * Add simple send and receive programs * speedups receiving + better way to wait for thing * receive speedup by using linked list of chunks instead of large array * document bug * use chunks to receive data * trailing spaces
2019-03-01 06:54:03 +01:00
// Simple chat program that talks to a broadcast server
// Broadcast server can be ran with `ws broadcast_server`
//
2019-09-23 19:25:23 +02:00
#include "nlohmann/json.hpp"
2020-01-09 22:45:31 +01:00
#include <iostream>
2018-10-09 06:42:45 +02:00
#include <ixwebsocket/IXSocket.h>
2019-09-23 19:25:23 +02:00
#include <ixwebsocket/IXWebSocket.h>
#include <queue>
#include <spdlog/spdlog.h>
2019-12-30 17:46:18 +01:00
#include <sstream>
2018-09-27 23:56:48 +02:00
// for convenience
using json = nlohmann::json;
namespace ix
2018-09-27 23:56:48 +02:00
{
class WebSocketChat
{
2019-09-23 19:25:23 +02:00
public:
WebSocketChat(const std::string& url, const std::string& user);
2018-09-27 23:56:48 +02:00
2019-09-23 19:25:23 +02:00
void subscribe(const std::string& channel);
void start();
void stop();
bool isReady() const;
2018-09-27 23:56:48 +02:00
2019-09-23 19:25:23 +02:00
void sendMessage(const std::string& text);
size_t getReceivedMessagesCount() const;
2018-09-27 23:56:48 +02:00
2019-09-23 19:25:23 +02:00
std::string encodeMessage(const std::string& text);
std::pair<std::string, std::string> decodeMessage(const std::string& str);
2018-09-27 23:56:48 +02:00
2019-09-23 19:25:23 +02:00
private:
std::string _url;
std::string _user;
ix::WebSocket _webSocket;
std::queue<std::string> _receivedQueue;
2019-09-23 19:25:23 +02:00
void log(const std::string& msg);
2018-09-27 23:56:48 +02:00
};
2019-09-23 19:25:23 +02:00
WebSocketChat::WebSocketChat(const std::string& url, const std::string& user)
: _url(url)
, _user(user)
2018-09-27 23:56:48 +02:00
{
;
}
void WebSocketChat::log(const std::string& msg)
{
spdlog::info(msg);
}
2018-09-27 23:56:48 +02:00
size_t WebSocketChat::getReceivedMessagesCount() const
{
return _receivedQueue.size();
}
bool WebSocketChat::isReady() const
{
return _webSocket.getReadyState() == ix::ReadyState::Open;
2018-09-27 23:56:48 +02:00
}
void WebSocketChat::stop()
{
_webSocket.stop();
}
void WebSocketChat::start()
{
_webSocket.setUrl(_url);
2018-09-27 23:56:48 +02:00
std::stringstream ss;
log(std::string("Connecting to url: ") + _url);
2018-09-27 23:56:48 +02:00
2019-09-23 19:25:23 +02:00
_webSocket.setOnMessageCallback([this](const WebSocketMessagePtr& msg) {
std::stringstream ss;
if (msg->type == ix::WebSocketMessageType::Open)
2018-09-27 23:56:48 +02:00
{
2019-09-23 19:25:23 +02:00
log("ws chat: connected");
spdlog::info("Uri: {}", msg->openInfo.uri);
spdlog::info("Headers:");
2019-09-23 19:25:23 +02:00
for (auto it : msg->openInfo.headers)
2018-09-27 23:56:48 +02:00
{
spdlog::info("{}: {}", it.first, it.second);
2018-09-27 23:56:48 +02:00
}
spdlog::info("ws chat: user {} connected !", _user);
2019-09-23 19:25:23 +02:00
log(ss.str());
}
else if (msg->type == ix::WebSocketMessageType::Close)
{
ss << "ws chat user disconnected: " << _user;
ss << " code " << msg->closeInfo.code;
ss << " reason " << msg->closeInfo.reason << std::endl;
2019-09-23 19:25:23 +02:00
log(ss.str());
}
else if (msg->type == ix::WebSocketMessageType::Message)
{
auto result = decodeMessage(msg->str);
2018-09-27 23:56:48 +02:00
2019-09-23 19:25:23 +02:00
// Our "chat" / "broacast" node.js server does not send us
// the messages we send, so we don't have to filter it out.
2018-09-27 23:56:48 +02:00
2019-09-23 19:25:23 +02:00
// store text
_receivedQueue.push(result.second);
ss << std::endl
<< result.first << "(" << msg->wireSize << " bytes)"
<< " > " << result.second << std::endl
<< _user << " > ";
log(ss.str());
}
else if (msg->type == ix::WebSocketMessageType::Error)
{
ss << "Connection error: " << msg->errorInfo.reason << std::endl;
ss << "#retries: " << msg->errorInfo.retries << std::endl;
ss << "Wait time(ms): " << msg->errorInfo.wait_time << std::endl;
ss << "HTTP Status: " << msg->errorInfo.http_status << std::endl;
log(ss.str());
}
else
{
ss << "Invalid ix::WebSocketMessageType";
log(ss.str());
}
});
2018-09-27 23:56:48 +02:00
_webSocket.start();
}
std::pair<std::string, std::string> WebSocketChat::decodeMessage(const std::string& str)
{
auto j = json::parse(str);
std::string msg_user = j["user"];
std::string msg_text = j["text"];
return std::pair<std::string, std::string>(msg_user, msg_text);
}
std::string WebSocketChat::encodeMessage(const std::string& text)
{
json j;
j["user"] = _user;
j["text"] = text;
std::string output = j.dump();
return output;
}
void WebSocketChat::sendMessage(const std::string& text)
{
2019-06-09 19:22:27 +02:00
_webSocket.sendText(encodeMessage(text));
2018-09-27 23:56:48 +02:00
}
2019-09-23 19:25:23 +02:00
int ws_chat_main(const std::string& url, const std::string& user)
2018-09-27 23:56:48 +02:00
{
spdlog::info("Type Ctrl-D to exit prompt...");
WebSocketChat webSocketChat(url, user);
2018-09-27 23:56:48 +02:00
webSocketChat.start();
while (true)
{
// Read line
std::string line;
2020-01-08 02:37:38 +01:00
std::cout << user << " > " << std::flush;
std::getline(std::cin, line);
2018-09-27 23:56:48 +02:00
2020-01-08 02:37:38 +01:00
if (!std::cin)
2018-09-27 23:56:48 +02:00
{
break;
}
webSocketChat.sendMessage(line);
2018-09-27 23:56:48 +02:00
}
spdlog::info("");
2018-09-27 23:56:48 +02:00
webSocketChat.stop();
return 0;
2018-09-27 23:56:48 +02:00
}
2019-09-23 19:25:23 +02:00
} // namespace ix