/***************************************************************************** * dtranslatebot Discord Translate Bot * Copyright (C) 2023-2024 Syping * * Redistribution and use in source and binary forms, with or without modification, * are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * This software is provided as-is, no warranties are given to you, we are not * responsible for anything with use of the software, you are self responsible. *****************************************************************************/ #include #include #include #include #include "message_queue.h" #include "settings.h" int main(int argc, char* argv[]) { if (argc != 2) { std::cout << "Usage: " << argv[0] << " [json]" << std::endl; return 0; } bot::settings::settings settings; if (!settings.parse(argv[1])) return 1; { std::vector languages; languages = settings.get_translator()->get_languages(); if (languages.empty()) { std::cerr << "Failed to initialise translateable languages" << std::endl; return 2; } } dpp::cluster bot(settings.get_token(), dpp::i_default_intents | dpp::i_message_content); bot.on_log(dpp::utility::cout_logger()); bot::submit_queue submit_queue; std::thread submit_queue_loop(&bot::submit_queue::run, &submit_queue, &bot); bot::message_queue message_queue; std::thread message_queue_loop(&bot::message_queue::run, &message_queue, &settings, &submit_queue); bot.on_message_create([&bot, &message_queue, &settings](const dpp::message_create_t &event) { if (event.msg.webhook_id) { const std::lock_guard guard(settings); // We will not translate messages from our own bot if (settings.is_translatebot(event.msg.webhook_id)) return; } // Same as before, just without the involvement of webhooks if (event.msg.author.id == bot.me.id) return; const std::lock_guard guard(settings); const bot::settings::channel *channel = settings.get_channel(event.msg.guild_id, event.msg.channel_id); if (channel) { bot::message message; message.id = event.msg.id; message.author = event.msg.member.get_nickname(); if (message.author.empty()) message.author = event.msg.author.global_name; message.avatar = event.msg.member.get_avatar_url(settings.get_avatar_size()); if (message.avatar.empty()) message.avatar = event.msg.author.get_avatar_url(settings.get_avatar_size()); message.message = event.msg.content; message.source = channel->source; message.targets = channel->targets; message_queue.add(message); } }); bot.on_slashcommand([&bot, &settings](const dpp::slashcommand_t &event) { if (event.command.get_command_name() == "translate" || event.command.get_command_name() == "translate_pref") { try { std::variant v_target; const std::string source = std::get(event.get_parameter("source")); const std::string target = std::get(event.get_parameter("target")); dpp::command_interaction interaction = event.command.get_command_interaction(); if (interaction.options[0].name == "channel") { v_target = event.command.get_resolved_channel( std::get(event.get_parameter("channel"))); } else if (interaction.options[0].name == "webhook") { v_target = dpp::webhook(std::get(event.get_parameter("webhook"))); } const std::vector languages = settings.get_translator()->get_languages(); std::ostringstream language_codes; bool source_valid = false, target_valid = false; for (const bot::translate::language &language : languages) { if (language.code == source) source_valid = true; if (language.code == target) target_valid = true; if (source_valid && target_valid) break; language_codes << " " << language.code; } if (source_valid && target_valid) { const std::lock_guard guard(settings); if (!settings.get_channel(event.command.guild_id, event.command.channel_id)) { if (dpp::channel *channel = std::get_if(&v_target)) { dpp::webhook webhook; webhook.channel_id = channel->id; webhook.guild_id = channel->guild_id; webhook.name = "Translate Bot Webhook <" + std::to_string(event.command.channel_id) + ":" + source + ":" + target + ">"; bot.create_webhook(webhook, [&bot, &settings, event, source, target](const dpp::confirmation_callback_t &callback) { if (callback.is_error()) { event.reply(dpp::message("Failed to generate webhook!\n" + callback.http_info.body).set_flags(dpp::m_ephemeral)); return; } const dpp::webhook webhook = callback.get(); bot::settings::channel s_channel; s_channel.id = event.command.channel_id; s_channel.source = source; bot::settings::target s_target; s_target.target = target; s_target.webhook = webhook; s_channel.targets.push_back(s_target); settings.lock(); settings.add_channel(s_channel, event.command.guild_id); settings.add_translatebot_webhook(webhook); settings.unlock(); event.reply(dpp::message("Channel will be now translated!").set_flags(dpp::m_ephemeral)); }); } else if (dpp::webhook *webhook = std::get_if(&v_target)) { bot::settings::channel s_channel; s_channel.id = event.command.channel_id; s_channel.source = source; bot::settings::target s_target; s_target.target = target; s_target.webhook = *webhook; s_channel.targets.push_back(s_target); settings.lock(); settings.add_channel(s_channel, event.command.guild_id); settings.add_translatebot_webhook(*webhook); settings.unlock(); event.reply(dpp::message("Channel will be now translated!").set_flags(dpp::m_ephemeral)); } } else { event.reply(dpp::message("The current channel is already being translated!").set_flags(dpp::m_ephemeral)); } } else if (!source_valid && !target_valid) { event.reply(dpp::message("Source and target languages are not valid!\nAvailable languages are:" + language_codes.str()).set_flags(dpp::m_ephemeral)); } else if (!source_valid) { event.reply(dpp::message("Source language is not valid!\nAvailable languages are:" + language_codes.str()).set_flags(dpp::m_ephemeral)); } else if (!target_valid) { event.reply(dpp::message("Target language is not valid!\nAvailable languages are:" + language_codes.str()).set_flags(dpp::m_ephemeral)); } } catch (const std::exception &exception) { std::cerr << "Failed to process command /" << event.command.get_command_name() << ": " << exception.what() << std::endl; event.reply(dpp::message("Failed to process command /" + event.command.get_command_name() + "\n" + exception.what()).set_flags(dpp::m_ephemeral)); } } }); bot.on_ready([&bot, &settings](const dpp::ready_t &event) { if (dpp::run_once()) { settings.lock(); const std::vector languages = settings.get_translator()->get_languages(); const std::vector preferred_languages = settings.get_preferred_languages(); settings.unlock(); std::vector commands; dpp::slashcommand command_translate("translate", "Translate current channel (ISO 639-1)", bot.me.id); command_translate.set_default_permissions(dpp::p_manage_webhooks); dpp::command_option channel_subcommand(dpp::co_sub_command, "channel", "Translate current channel to a channel (ISO 639-1)"); dpp::command_option webhook_subcommand(dpp::co_sub_command, "webhook", "Translate current channel to a webhook (ISO 639-1)"); dpp::command_option source_option(dpp::co_string, "source", "Source language", true); source_option.set_max_length(2).set_min_length(2); dpp::command_option target_option(dpp::co_string, "target", "Target language", true); target_option.set_max_length(2).set_min_length(2); dpp::command_option channel_option(dpp::co_channel, "channel", "Target channel", true); channel_option.add_channel_type(dpp::CHANNEL_TEXT); dpp::command_option webhook_option(dpp::co_string, "webhook", "Target webhook", true); channel_subcommand.add_option(source_option); channel_subcommand.add_option(target_option); channel_subcommand.add_option(channel_option); webhook_subcommand.add_option(source_option); webhook_subcommand.add_option(target_option); webhook_subcommand.add_option(webhook_option); command_translate.add_option(channel_subcommand); command_translate.add_option(webhook_subcommand); commands.push_back(command_translate); if (preferred_languages.size() > 1) { dpp::slashcommand command_translate_pref("translate_pref", "Translate current channel (Preferred languages)", bot.me.id); command_translate_pref.set_default_permissions(dpp::p_manage_webhooks); dpp::command_option channel_pref_subcommand(dpp::co_sub_command, "channel", "Translate current channel to a channel (Preferred languages)"); dpp::command_option webhook_pref_subcommand(dpp::co_sub_command, "webhook", "Translate current channel to a webhook (Preferred languages)"); dpp::command_option source_pref_option(dpp::co_string, "source", "Source language", true); dpp::command_option target_pref_option(dpp::co_string, "target", "Target language", true); for (const bot::translate::language &language : languages) { if (std::find(preferred_languages.begin(), preferred_languages.end(), language.code) != preferred_languages.end()) { source_pref_option.add_choice(dpp::command_option_choice(language.name, language.code)); target_pref_option.add_choice(dpp::command_option_choice(language.name, language.code)); } } channel_pref_subcommand.add_option(source_pref_option); channel_pref_subcommand.add_option(target_pref_option); channel_pref_subcommand.add_option(channel_option); webhook_pref_subcommand.add_option(source_pref_option); webhook_pref_subcommand.add_option(target_pref_option); webhook_pref_subcommand.add_option(webhook_option); command_translate_pref.add_option(channel_pref_subcommand); command_translate_pref.add_option(webhook_pref_subcommand); commands.push_back(command_translate_pref); } bot.global_bulk_command_create(commands); } }); bot.start(dpp::st_wait); // It's unneccessary, but we choose to exit clean anyway message_queue.terminate(); message_queue_loop.join(); submit_queue.terminate(); submit_queue_loop.join(); return 0; }