/* * Copyright (C) 2017 Ortega Froysa, Nicolás * Author: Ortega Froysa, Nicolás * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as * published by the Free Software Foundation, either version 3 of the * License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see . */ #include "neocomm.h" #include "error.hpp" #include "globals.hpp" std::map channels; static std::mt19937_64 rand_dev {dht::crypto::random_device{}()}; static std::uniform_int_distribution rand_id; int NeoComm_join_channel(const char *channel_name) { if(not node.isRunning()) { add_error("NeoComm must be initialized."); return 0; } const std::string s_chan_name(channel_name); if(s_chan_name.empty()) { add_error("Chan name is empty."); return 0; } // if it already exists then just exit if(channels.find(s_chan_name) == channels.end()) return 1; dht::InfoHash chan_hash(s_chan_name); { static constexpr dht::InfoHash INVALID_ID {}; if(chan_hash == INVALID_ID) chan_hash = dht::InfoHash::get(s_chan_name); } channels[s_chan_name] = { /*.hash =*/ chan_hash, /*.token =*/ node.listen(chan_hash, [=](dht::ImMessage &&msg) { channels[s_chan_name].msgs.push_back(msg); return true; }), /*.msgs = */ std::vector() }; return 1; } void NeoComm_leave_channel(const char *channel_name) { if(not node.isRunning()) return; const std::string s_chan_name(channel_name); if(channels.find(s_chan_name) == channels.end()) return; struct channel_info *channel = &channels[s_chan_name]; channel->token.wait(); node.cancelListen(channel->hash, channel->token.get()); channels.erase(s_chan_name); } struct message *NeoComm_get_next_message(const char *channel_name) { const std::string s_chan_name(channel_name); if(channels[s_chan_name].msgs.empty()) return NULL; const dht::ImMessage *new_msg = &channels[s_chan_name].msgs.front(); struct message *msg = static_cast(malloc(sizeof(struct message))); *msg = { /*.msg =*/ new_msg->msg.c_str(), /*.sent =*/ new_msg->date, // TODO: add the rest of these }; return msg; } void NeoComm_free_message(struct message *msg) { free(msg); } int NeoComm_send_message(const char *channel_name, const char *message) { const dht::InfoHash chan_hash = channels[channel_name].hash; const time_t now = std::chrono::system_clock::to_time_t(std::chrono::system_clock::now()); if(channels.find(channel_name) == channels.end()) { add_error("Not connected to channel"); return 0; } node.putSigned(chan_hash, dht::ImMessage(rand_id(rand_dev), message, now), [](bool sent) { // TODO: Figure out what to do here for failed messages. }); return 1; }