diff options
Diffstat (limited to 'discord/discord.cpp')
-rw-r--r-- | discord/discord.cpp | 66 |
1 files changed, 62 insertions, 4 deletions
diff --git a/discord/discord.cpp b/discord/discord.cpp index cd1e723..42c6550 100644 --- a/discord/discord.cpp +++ b/discord/discord.cpp @@ -1,10 +1,18 @@ +#include "../abaddon.hpp" #include "discord.hpp" +#include <cassert> -DiscordClient::DiscordClient() {} +DiscordClient::DiscordClient() { + LoadEventMap(); +} + +void DiscordClient::SetAbaddon(Abaddon *ptr) { + m_abaddon = ptr; +} void DiscordClient::Start() { - if (m_client_connected) - throw std::runtime_error("attempt to start client twice consecutively"); + assert(!m_client_connected); + assert(!m_websocket.IsOpen()); m_client_connected = true; m_websocket.StartConnection(DiscordGateway); @@ -13,9 +21,11 @@ void DiscordClient::Start() { void DiscordClient::Stop() { if (!m_client_connected) return; + m_heartbeat_waiter.kill(); m_heartbeat_thread.join(); m_client_connected = false; + m_websocket.Stop(); } bool DiscordClient::IsStarted() const { @@ -36,16 +46,33 @@ void DiscordClient::HandleGatewayMessage(nlohmann::json j) { HelloMessageData d = m.Data; m_heartbeat_msec = d.HeartbeatInterval; m_heartbeat_thread = std::thread(std::bind(&DiscordClient::HeartbeatThread, this)); + SendIdentify(); } break; case GatewayOp::HeartbeatAck: { m_heartbeat_acked = true; } break; + case GatewayOp::Event: { + auto iter = m_event_map.find(m.Type); + if (iter == m_event_map.end()) { + printf("Unknown event %s\n", m.Type.c_str()); + break; + } + switch (iter->second) { + case GatewayEvent::READY: { + HandleGatewayReady(m); + } + } + } break; default: printf("Unknown opcode %d\n", m.Opcode); break; } } +void DiscordClient::HandleGatewayReady(const GatewayMessage &msg) { + +} + void DiscordClient::HeartbeatThread() { while (m_client_connected) { if (!m_heartbeat_acked) { @@ -57,13 +84,28 @@ void DiscordClient::HeartbeatThread() { HeartbeatMessage msg; msg.Sequence = m_last_sequence; nlohmann::json j = msg; - m_websocket.Send(j.dump()); + m_websocket.Send(j); if (!m_heartbeat_waiter.wait_for(std::chrono::milliseconds(m_heartbeat_msec))) break; } } +void DiscordClient::SendIdentify() { + auto token = m_abaddon->GetDiscordToken(); + assert(token.size()); + IdentifyMessage msg; + msg.Properties.OS = "OpenBSD"; + msg.Properties.Device = GatewayIdentity; + msg.Properties.Browser = GatewayIdentity; + msg.Token = token; + m_websocket.Send(msg); +} + +void DiscordClient::LoadEventMap() { + m_event_map["READY"] = GatewayEvent::READY; +} + void from_json(const nlohmann::json &j, GatewayMessage &m) { j.at("op").get_to(m.Opcode); m.Data = j.at("d"); @@ -76,6 +118,22 @@ void from_json(const nlohmann::json &j, HelloMessageData &m) { j.at("heartbeat_interval").get_to(m.HeartbeatInterval); } +void to_json(nlohmann::json &j, const IdentifyProperties &m) { + j["$os"] = m.OS; + j["$browser"] = m.Browser; + j["$device"] = m.Device; +} + +void to_json(nlohmann::json &j, const IdentifyMessage &m) { + j["op"] = GatewayOp::Identify; + j["d"] = nlohmann::json::object(); + j["d"]["token"] = m.Token; + j["d"]["properties"] = m.Properties; + + if (m.LargeThreshold) + j["d"]["large_threshold"] = m.LargeThreshold; +} + void to_json(nlohmann::json &j, const HeartbeatMessage &m) { j["op"] = GatewayOp::Heartbeat; if (m.Sequence == -1) |