2023-12-01 02:48:18 +01:00
|
|
|
#include <solanaceae/util/simple_config_model.hpp>
|
|
|
|
#include <solanaceae/contact/contact_model3.hpp>
|
|
|
|
#include <solanaceae/message3/registry_message_model.hpp>
|
|
|
|
#include <solanaceae/message3/message_time_sort.hpp>
|
|
|
|
#include <solanaceae/plugin/plugin_manager.hpp>
|
|
|
|
#include <solanaceae/toxcore/tox_event_logger.hpp>
|
|
|
|
#include "./tox_private_impl.hpp"
|
|
|
|
|
|
|
|
#include <solanaceae/tox_contacts/tox_contact_model2.hpp>
|
|
|
|
#include <solanaceae/tox_messages/tox_message_manager.hpp>
|
|
|
|
#include <solanaceae/tox_messages/tox_transfer_manager.hpp>
|
|
|
|
|
|
|
|
#include "./tox_client.hpp"
|
|
|
|
#include "./auto_dirty.hpp"
|
2023-12-02 02:55:26 +01:00
|
|
|
#include "./message_cleanser.hpp"
|
2023-12-03 16:01:33 +01:00
|
|
|
#include "./message_command_dispatcher.hpp"
|
2023-12-01 02:48:18 +01:00
|
|
|
|
2023-12-03 20:01:47 +01:00
|
|
|
#include <solanaceae/message3/components.hpp>
|
|
|
|
#include <solanaceae/contact/components.hpp>
|
|
|
|
#include <solanaceae/tox_contacts/components.hpp>
|
|
|
|
#include <solanaceae/toxcore/utils.hpp>
|
|
|
|
|
2023-12-01 19:26:46 +01:00
|
|
|
#include <nlohmann/json.hpp>
|
|
|
|
|
2023-12-01 02:07:26 +01:00
|
|
|
#include <chrono>
|
|
|
|
#include <thread>
|
|
|
|
#include <atomic>
|
|
|
|
#include <iostream>
|
2023-12-01 19:26:46 +01:00
|
|
|
#include <fstream>
|
|
|
|
#include <filesystem>
|
2023-12-01 02:07:26 +01:00
|
|
|
|
|
|
|
#if defined (__unix__) || (defined (__APPLE__) && defined (__MACH__))
|
|
|
|
#include <signal.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#elif defined (_WIN32)
|
|
|
|
#include <signal.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
std::atomic_bool quit = false;
|
|
|
|
|
|
|
|
#if defined (__unix__) || (defined (__APPLE__) && defined (__MACH__)) || defined (_WIN32)
|
|
|
|
void sigint_handler(int signo) {
|
|
|
|
if (signo == SIGINT) {
|
|
|
|
quit = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2023-12-01 19:26:46 +01:00
|
|
|
bool load_json_into_config(const nlohmann::ordered_json& config_json, SimpleConfigModel& conf) {
|
|
|
|
if (!config_json.is_object()) {
|
|
|
|
std::cout << "TOTATO error: config file is not an json object!!!\n";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
for (const auto& [mod, cats] : config_json.items()) {
|
|
|
|
for (const auto& [cat, cat_v] : cats.items()) {
|
|
|
|
if (cat_v.is_object()) {
|
|
|
|
if (cat_v.contains("default")) {
|
|
|
|
const auto& value = cat_v["default"];
|
|
|
|
if (value.is_string()) {
|
|
|
|
conf.set(mod, cat, value.get_ref<const std::string&>());
|
|
|
|
} else if (value.is_boolean()) {
|
|
|
|
conf.set(mod, cat, value.get_ref<const bool&>());
|
|
|
|
} else if (value.is_number_float()) {
|
|
|
|
conf.set(mod, cat, value.get_ref<const double&>());
|
|
|
|
} else if (value.is_number_integer()) {
|
|
|
|
conf.set(mod, cat, value.get_ref<const int64_t&>());
|
|
|
|
} else {
|
|
|
|
std::cerr << "JSON error: wrong value type in " << mod << "::" << cat << " = " << value << "\n";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (cat_v.contains("entries")) {
|
|
|
|
for (const auto& [ent, ent_v] : cat_v["entries"].items()) {
|
|
|
|
if (ent_v.is_string()) {
|
|
|
|
conf.set(mod, cat, ent, ent_v.get_ref<const std::string&>());
|
|
|
|
} else if (ent_v.is_boolean()) {
|
|
|
|
conf.set(mod, cat, ent, ent_v.get_ref<const bool&>());
|
|
|
|
} else if (ent_v.is_number_float()) {
|
|
|
|
conf.set(mod, cat, ent, ent_v.get_ref<const double&>());
|
|
|
|
} else if (ent_v.is_number_integer()) {
|
|
|
|
conf.set(mod, cat, ent, ent_v.get_ref<const int64_t&>());
|
|
|
|
} else {
|
|
|
|
std::cerr << "JSON error: wrong value type in " << mod << "::" << cat << "::" << ent << " = " << ent_v << "\n";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (cat_v.is_string()) {
|
|
|
|
conf.set(mod, cat, cat_v.get_ref<const std::string&>());
|
|
|
|
} else if (cat_v.is_boolean()) {
|
|
|
|
conf.set(mod, cat, cat_v.get_ref<const bool&>());
|
|
|
|
} else if (cat_v.is_number_float()) {
|
|
|
|
conf.set(mod, cat, cat_v.get_ref<const double&>());
|
|
|
|
} else if (cat_v.is_number_integer()) {
|
|
|
|
conf.set(mod, cat, cat_v.get_ref<const int64_t&>());
|
|
|
|
} else {
|
|
|
|
std::cerr << "JSON error: wrong value type in " << mod << "::" << cat << " = " << cat_v << "\n";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-12-01 02:07:26 +01:00
|
|
|
int main(int argc, char** argv) {
|
|
|
|
#if defined (__unix__) || (defined (__APPLE__) && defined (__MACH__))
|
|
|
|
struct sigaction sigint_action;
|
|
|
|
sigint_action.sa_handler = sigint_handler;
|
|
|
|
sigemptyset (&sigint_action.sa_mask);
|
|
|
|
sigint_action.sa_flags = 0;
|
|
|
|
sigaction(SIGINT, &sigint_action, NULL);
|
|
|
|
#elif defined (_WIN32)
|
|
|
|
signal(SIGINT, sigint_handler);
|
|
|
|
#endif
|
2023-12-01 19:26:46 +01:00
|
|
|
|
2023-12-01 02:48:18 +01:00
|
|
|
auto last_time = std::chrono::steady_clock::now();
|
2023-12-01 02:07:26 +01:00
|
|
|
|
2023-12-01 19:26:46 +01:00
|
|
|
std::string config_path {"config.json"};
|
2023-12-01 02:07:26 +01:00
|
|
|
|
2023-12-01 19:26:46 +01:00
|
|
|
// totato <config.json> -p <path/to/plugin.so>
|
|
|
|
// TODO: parse arg
|
|
|
|
if (argc == 2) {
|
|
|
|
config_path = argv[1];
|
|
|
|
}
|
2023-12-01 02:48:18 +01:00
|
|
|
|
|
|
|
SimpleConfigModel conf;
|
2023-12-01 19:26:46 +01:00
|
|
|
|
|
|
|
{ // read conf from json TODO: refactor extract this for reuse
|
|
|
|
auto config_file = std::ifstream(config_path);
|
|
|
|
if (!config_file.is_open()) {
|
|
|
|
std::cerr << "TOTATO error: failed to open config file '" << config_path << "', exiting...\n";
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto config_json = nlohmann::ordered_json::parse(std::ifstream(config_path));
|
|
|
|
if (!load_json_into_config(config_json, conf)) {
|
|
|
|
std::cerr << "TOTATO error in config json, exiting...\n";
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{ // fill in defaults
|
|
|
|
// config file folder
|
|
|
|
const auto config_path_base = std::filesystem::path{config_path}.parent_path();
|
|
|
|
|
|
|
|
if (!conf.has_string("tox", "save_file_path")) {
|
|
|
|
// default to totato.tox relative to config file
|
|
|
|
conf.set("tox", "save_file_path", (config_path_base / "totato.tox").u8string());
|
|
|
|
} else { // transform relative to config to absolute
|
2023-12-02 03:26:01 +01:00
|
|
|
const auto tox_conf_path = std::filesystem::path{static_cast<std::string_view>(conf.get_string("tox", "save_file_path").value())};
|
2023-12-02 00:17:37 +01:00
|
|
|
if (tox_conf_path.is_relative()) {
|
2023-12-01 19:26:46 +01:00
|
|
|
// is relative to config
|
2023-12-02 00:17:37 +01:00
|
|
|
conf.set("tox", "save_file_path", std::filesystem::canonical(config_path_base / tox_conf_path).u8string());
|
2023-12-01 19:26:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: name
|
|
|
|
}
|
|
|
|
|
2023-12-01 02:48:18 +01:00
|
|
|
Contact3Registry cr;
|
|
|
|
RegistryMessageModel rmm{cr};
|
|
|
|
MessageTimeSort mts{rmm};
|
2023-12-02 02:55:26 +01:00
|
|
|
MessageCleanser mc{cr, rmm};
|
2023-12-03 16:01:33 +01:00
|
|
|
MessageCommandDispatcher mcd{cr, rmm, conf};
|
|
|
|
|
|
|
|
{ // setup basic commands for bot
|
|
|
|
mcd.registerCommand(
|
|
|
|
"host", "",
|
|
|
|
"info",
|
2023-12-03 20:01:47 +01:00
|
|
|
[](std::string_view, Message3Handle m) -> bool {
|
2023-12-03 16:01:33 +01:00
|
|
|
std::cout << "INFO got called :)\n";
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
"Get basic information about this bot"
|
|
|
|
);
|
2023-12-03 20:01:47 +01:00
|
|
|
|
2023-12-03 16:01:33 +01:00
|
|
|
}
|
2023-12-01 02:48:18 +01:00
|
|
|
|
|
|
|
PluginManager pm;
|
|
|
|
|
2023-12-01 19:26:46 +01:00
|
|
|
ToxEventLogger tel{std::cout}; // TODO: config
|
|
|
|
|
|
|
|
// TODO: password?
|
|
|
|
ToxClient tc{conf.get_string("tox", "save_file_path").value(), ""};
|
|
|
|
tel.subscribeAll(tc);
|
|
|
|
{ // name stuff
|
|
|
|
auto name = tc.toxSelfGetName();
|
|
|
|
if (name.empty()) {
|
|
|
|
name = "totato";
|
|
|
|
}
|
|
|
|
conf.set("tox", "name", name);
|
|
|
|
tc.setSelfName(name); // TODO: this is ugly
|
|
|
|
}
|
|
|
|
|
2023-12-02 02:55:26 +01:00
|
|
|
std::cout << "TOTATO: own address: " << tc.toxSelfGetAddressStr() << "\n";
|
|
|
|
|
2023-12-01 02:48:18 +01:00
|
|
|
ToxPrivateImpl tpi{tc.getTox()};
|
|
|
|
AutoDirty ad{tc};
|
|
|
|
ToxContactModel2 tcm{cr, tc, tc};
|
|
|
|
ToxMessageManager tmm{rmm, cr, tcm, tc, tc};
|
|
|
|
ToxTransferManager ttm{rmm, cr, tcm, tc, tc};
|
|
|
|
|
2023-12-03 20:01:47 +01:00
|
|
|
{ // setup more advanced commands
|
|
|
|
mcd.registerCommand(
|
|
|
|
"tox", "tox",
|
|
|
|
"status",
|
|
|
|
[&](std::string_view, Message3Handle m) -> bool {
|
|
|
|
const auto tox_self_status = tc.toxSelfGetConnectionStatus();
|
|
|
|
|
|
|
|
const auto contact_from = m.get<Message::Components::ContactFrom>().c;
|
|
|
|
|
|
|
|
std::string reply{"dht:"};
|
|
|
|
|
|
|
|
if (tox_self_status == Tox_Connection::TOX_CONNECTION_UDP) {
|
|
|
|
reply += "upd-direct";
|
|
|
|
} else if (tox_self_status == Tox_Connection::TOX_CONNECTION_TCP) {
|
|
|
|
reply += "tcp-relayed";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cr.all_of<Contact::Components::ToxFriendEphemeral>(contact_from)) {
|
|
|
|
const auto con_opt = tc.toxFriendGetConnectionStatus(cr.get<Contact::Components::ToxFriendEphemeral>(contact_from).friend_number);
|
|
|
|
if (!con_opt.has_value() || con_opt.value() == Tox_Connection::TOX_CONNECTION_NONE) {
|
|
|
|
reply += "\nfriend:offline";
|
|
|
|
} else if (con_opt.value() == Tox_Connection::TOX_CONNECTION_UDP) {
|
|
|
|
reply += "\nfriend:udp-direct";
|
|
|
|
} else if (con_opt.value() == Tox_Connection::TOX_CONNECTION_TCP) {
|
|
|
|
reply += "\nfriend:tcp-relayed";
|
|
|
|
}
|
|
|
|
} else if (cr.all_of<Contact::Components::ToxGroupPeerEphemeral>(contact_from)) {
|
|
|
|
const auto [group_number, peer_number] = cr.get<Contact::Components::ToxGroupPeerEphemeral>(contact_from);
|
|
|
|
|
|
|
|
const auto [con_opt, _] = tc.toxGroupPeerGetConnectionStatus(group_number, peer_number);
|
|
|
|
if (!con_opt.has_value() || con_opt.value() == Tox_Connection::TOX_CONNECTION_NONE) {
|
|
|
|
reply += "\ngroup-peer:offline";
|
|
|
|
} else if (con_opt.value() == Tox_Connection::TOX_CONNECTION_UDP) {
|
|
|
|
reply += "\ngroup-peer:udp-direct";
|
|
|
|
} else if (con_opt.value() == Tox_Connection::TOX_CONNECTION_TCP) {
|
|
|
|
reply += "\ngroup-peer:tcp-relayed";
|
|
|
|
}
|
|
|
|
} else if (cr.any_of<Contact::Components::ToxFriendPersistent, Contact::Components::ToxGroupPeerPersistent>(contact_from)) {
|
|
|
|
reply += "\noffline";
|
|
|
|
} else {
|
|
|
|
reply += "\nunk";
|
|
|
|
}
|
|
|
|
|
|
|
|
rmm.sendText(
|
|
|
|
contact_from,
|
|
|
|
reply
|
|
|
|
);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
"Query the tox status of dht and to you."
|
|
|
|
);
|
|
|
|
|
|
|
|
mcd.registerCommand(
|
|
|
|
"tox", "tox",
|
|
|
|
"add",
|
|
|
|
[&](std::string_view params, Message3Handle m) -> bool {
|
|
|
|
const auto contact_from = m.get<Message::Components::ContactFrom>().c;
|
|
|
|
|
|
|
|
if (params.size() != 38*2) {
|
|
|
|
rmm.sendText(
|
|
|
|
contact_from,
|
|
|
|
"error adding friend, id is not the correct size!"
|
|
|
|
);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: add tcm interface
|
|
|
|
const auto [_, err] = tc.toxFriendAdd(hex2bin(std::string{params}), "Add me, I am totato");
|
|
|
|
|
|
|
|
if (err == Tox_Err_Friend_Add::TOX_ERR_FRIEND_ADD_OK) {
|
|
|
|
rmm.sendText(
|
|
|
|
contact_from,
|
|
|
|
"freind request sent"
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
rmm.sendText(
|
|
|
|
contact_from,
|
|
|
|
"error adding friend, error code " + std::to_string(err)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
"add a tox friend by id"
|
|
|
|
);
|
|
|
|
|
|
|
|
mcd.registerCommand(
|
|
|
|
"tox", "tox",
|
|
|
|
"join",
|
|
|
|
[&](std::string_view params, Message3Handle m) -> bool {
|
|
|
|
const auto contact_from = m.get<Message::Components::ContactFrom>().c;
|
|
|
|
|
|
|
|
if (params.size() != 32*2) {
|
|
|
|
rmm.sendText(
|
|
|
|
contact_from,
|
|
|
|
"error joining group, id is not the correct size!"
|
|
|
|
);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto name_opt = conf.get_string("tox", "name");
|
|
|
|
|
|
|
|
// TODO: add tcm interface
|
|
|
|
const auto [_, err] = tc.toxGroupJoin(hex2bin(std::string{params}), std::string{name_opt.value_or("no-name-found")}, "");
|
|
|
|
if (err == Tox_Err_Group_Join::TOX_ERR_GROUP_JOIN_OK) {
|
|
|
|
rmm.sendText(
|
|
|
|
contact_from,
|
|
|
|
"joining group..."
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
rmm.sendText(
|
|
|
|
contact_from,
|
|
|
|
"error joining group, error code " + std::to_string(err)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
"join a tox group by id"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-12-01 02:48:18 +01:00
|
|
|
{ // setup plugin instances
|
|
|
|
g_provideInstance<ConfigModelI>("ConfigModelI", "host", &conf);
|
|
|
|
g_provideInstance<Contact3Registry>("Contact3Registry", "host", &cr);
|
|
|
|
g_provideInstance<RegistryMessageModel>("RegistryMessageModel", "host", &rmm);
|
2023-12-03 20:01:47 +01:00
|
|
|
g_provideInstance<MessageCommandDispatcher>("MessageCommandDispatcher", "host", &mcd);
|
2023-12-01 02:48:18 +01:00
|
|
|
|
2023-12-01 19:26:46 +01:00
|
|
|
g_provideInstance<ToxI>("ToxI", "host", &tc);
|
|
|
|
g_provideInstance<ToxPrivateI>("ToxPrivateI", "host", &tpi);
|
|
|
|
g_provideInstance<ToxEventProviderI>("ToxEventProviderI", "host", &tc);
|
|
|
|
g_provideInstance<ToxContactModel2>("ToxContactModel2", "host", &tcm);
|
2023-12-01 02:48:18 +01:00
|
|
|
|
|
|
|
// TODO: pm?
|
|
|
|
}
|
|
|
|
|
2023-12-01 19:26:46 +01:00
|
|
|
// load from config!!!
|
2023-12-02 00:17:37 +01:00
|
|
|
// HACK: we cheat and directly access the members
|
|
|
|
// TODO: add api to iterate
|
|
|
|
if (conf._map_bool.count("PluginManager") && conf._map_bool.at("PluginManager").count("autoload")) {
|
|
|
|
const auto config_path_base = std::filesystem::path{config_path}.parent_path();
|
|
|
|
|
|
|
|
for (const auto& [plugin_path, plugin_autoload] : conf._map_bool.at("PluginManager").at("autoload").second) {
|
|
|
|
if (plugin_autoload) {
|
|
|
|
std::filesystem::path real_plugin_path = plugin_path;
|
|
|
|
if (real_plugin_path.is_relative()) {
|
|
|
|
real_plugin_path = config_path_base / real_plugin_path;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!pm.add(real_plugin_path.u8string())) {
|
|
|
|
std::cerr << "TOTATO error: loading plugin '" << real_plugin_path << "' failed!\n";
|
|
|
|
// thow?
|
|
|
|
assert(false && "failed to load plugin");
|
|
|
|
}
|
|
|
|
}
|
2023-12-01 02:48:18 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
conf.dump();
|
2023-12-01 02:07:26 +01:00
|
|
|
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(10)); // at startup, just to be safe
|
|
|
|
|
|
|
|
while (!quit) {
|
|
|
|
//auto new_time = std::chrono::steady_clock::now();
|
|
|
|
|
2023-12-01 02:48:18 +01:00
|
|
|
quit = !tc.iterate();
|
|
|
|
tcm.iterate(/*time_delta*/0.02f);
|
|
|
|
ttm.iterate();
|
|
|
|
|
|
|
|
mts.iterate();
|
|
|
|
|
|
|
|
pm.tick(/*time_delta*/0.02f);
|
|
|
|
|
2023-12-02 02:55:26 +01:00
|
|
|
mc.iterate(0.02f);
|
|
|
|
|
2023-12-04 02:16:37 +01:00
|
|
|
mcd.iterate(0.02f);
|
|
|
|
|
2023-12-01 02:07:26 +01:00
|
|
|
//std::this_thread::sleep_for( // time left to get to 60fps
|
|
|
|
//std::chrono::duration<float, std::chrono::seconds::period>(0.0166f) // 60fps frame duration
|
|
|
|
//- std::chrono::duration<float, std::chrono::seconds::period>(std::chrono::steady_clock::now() - new_time) // time used for rendering
|
|
|
|
//);
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(20)); // HACK: until i figure out the best main loop
|
|
|
|
}
|
|
|
|
|
|
|
|
std::cout << "\nTOTATO shutting down...\n";
|
|
|
|
|
2023-12-01 00:24:18 +01:00
|
|
|
return 0;
|
|
|
|
}
|
2023-12-01 02:07:26 +01:00
|
|
|
|