shortener-bot/commands.cpp
2024-10-14 01:31:03 +03:00

192 lines
7.9 KiB
C++

#include "commands.h"
#include "common.h"
#include "curl/curl.h"
#include "curl/easy.h"
#include "td/telegram/td_api.h"
#include "td/tl/TlObject.h"
#include <cstring>
#include <functional>
#include <random>
#include <spdlog/spdlog.h>
#include <stdexcept>
#include <string>
#include <string_view>
#include <utility>
#include <vector>
void cmd::handle_regular_message(context *ctx, td_api::message &msg) {
if (msg.content_->get_id() == td_api::messageText::ID) {
std::string text = static_cast<td_api::messageText&>(*msg.content_).text_->text_;
if (std::strncmp(text.c_str(), "/shorten ", 9) == 0) {
std::string_view param(text.begin() + 9, text.end());
size_t nextSpace = param.find(' ');
if (nextSpace != std::string_view::npos)
param.remove_suffix(param.size() - nextSpace);
spdlog::info("Command /shorten received with parameter '{}'", param);
// TODO check URL validity
bool result = shorten_link(std::string(param), ctx, [tg = ctx->tg, chat_id = msg.chat_id_, thread_id = msg.message_thread_id_](std::string url){
tg->send_query(td_api::make_object<td_api::sendMessage>(
chat_id, thread_id,
nullptr /*reply_to*/, nullptr /*options*/,
nullptr, // reply_markup
static_cast<td_api::object_ptr<td_api::InputMessageContent>>(td_api::make_object<td_api::inputMessageText>(
std::move(td_api::make_object<td_api::formattedText>(url, std::move(std::vector<td_api::object_ptr<td_api::textEntity>>()))),
nullptr /*link_preview_options*/, false /*clear_draft*/
))
), {});
});
if (!result) {
ctx->tg->send_query(td_api::make_object<td_api::sendMessage>(
msg.chat_id_, msg.message_thread_id_,
nullptr /*reply_to*/, nullptr /*options*/,
nullptr, // reply_markup
static_cast<td_api::object_ptr<td_api::InputMessageContent>>(td_api::make_object<td_api::inputMessageText>(
std::move(td_api::make_object<td_api::formattedText>("произошла какая-то ошибка :(", std::move(std::vector<td_api::object_ptr<td_api::textEntity>>()))),
nullptr /*link_preview_options*/, false /*clear_draft*/
))
), {});
}
} else if (std::strncmp(text.c_str(), "/shorten", 8) == 0) {
std::string textRaw("usage: /shorten <url>");
std::vector<td_api::object_ptr<td_api::textEntity>> empty;
auto text = static_cast<td_api::object_ptr<td_api::InputMessageContent>>(td_api::make_object<td_api::inputMessageText>(td_api::make_object<td_api::formattedText>(textRaw, std::move(empty)), nullptr, false));
ctx->tg->send_query(td_api::make_object<td_api::sendMessage>(msg.chat_id_, msg.message_thread_id_, nullptr, nullptr, nullptr, std::move(text)), {});
}
}
}
void cmd::handle_inline_query(context *ctx, td_api::updateNewInlineQuery &query) {
// TODO check URL validity
uint64_t pendingQueryId = new_pending_query_id();
ctx->inlineQueries.insert(std::pair<uint64_t, pending_inline_query>(pendingQueryId, {query.query_, std::chrono::steady_clock::now() + std::chrono::hours(5)}));
std::vector<td_api::object_ptr<td_api::InputInlineQueryResult>> results;
results.reserve(1);
auto messageButton = td_api::make_object<td_api::inlineKeyboardButton>(
"press to shorten",
static_cast<td_api::object_ptr<td_api::InlineKeyboardButtonType>>(td_api::make_object<td_api::inlineKeyboardButtonTypeCallback>(std::to_string(pendingQueryId)))
);
std::vector<decltype(messageButton)> messageButtonRow;
messageButtonRow.push_back(std::move(messageButton));
std::vector<decltype(messageButtonRow)> messageButtonRows;
messageButtonRows.push_back(std::move(messageButtonRow));
results.push_back(static_cast<td_api::object_ptr<td_api::InputInlineQueryResult>>(td_api::make_object<td_api::inputInlineQueryResultArticle>(
"shorten",
"",
true, // hide_url
"Shorten!",
"...",
"https://slavasil.ru/favicon.ico",
48, 48,
static_cast<td_api::object_ptr<td_api::ReplyMarkup>>(td_api::make_object<td_api::replyMarkupInlineKeyboard>(
std::move(messageButtonRows)
)),
static_cast<td_api::object_ptr<td_api::InputMessageContent>>(td_api::make_object<td_api::inputMessageText>(
td_api::make_object<td_api::formattedText>(
query.query_,
std::move(std::vector<td_api::object_ptr<td_api::textEntity>>())
),
td_api::make_object<td_api::linkPreviewOptions>(true, "", false, false, false),
false // clear_draft
))
)));
ctx->tg->send_query(td_api::make_object<td_api::answerInlineQuery>(
query.id_,
false, // is_personal
nullptr, // button
std::move(results),
600, // cache_time
"" // next_offset
), {});
}
void cmd::handle_callback_query(context *ctx, td_api::updateNewInlineCallbackQuery &query) {
spdlog::info("callback query");
if (query.payload_->get_id() != td_api::callbackQueryPayloadData::ID) return;
auto payload = td::move_tl_object_as<td_api::callbackQueryPayloadData>(query.payload_);
ctx->tg->send_query(td_api::make_object<td_api::answerCallbackQuery>(
query.id_,
payload->data_,
false,
"",
0
), [](TgObject obj){
if (obj->get_id() == td_api::error::ID) {
spdlog::error("answerCallbackQuery error: {}", static_cast<td_api::error*>(obj.get())->message_);
}
});
uint64_t queryId = std::stoull(payload->data_);
spdlog::debug("is query in cache? {}", ctx->inlineQueries.contains(queryId));
try {
pending_inline_query &queryInfo = ctx->inlineQueries.at(queryId);
shorten_link(queryInfo.text, ctx, [ctx, inlineMessageId = query.inline_message_id_](std::string url){
ctx->tg->send_query(td_api::make_object<td_api::editInlineMessageText>(
inlineMessageId,
nullptr, // reply_markup
static_cast<td_api::object_ptr<td_api::InputMessageContent>>(td_api::make_object<td_api::inputMessageText>(
td_api::make_object<td_api::formattedText>(
url,
std::move(std::vector<td_api::object_ptr<td_api::textEntity>>())
),
td_api::make_object<td_api::linkPreviewOptions>(true, "", false, false, false),
//nullptr,
false
))
), {});
});
ctx->inlineQueries.erase(queryId);
} catch (std::out_of_range) {
spdlog::warn("a user tried to use an expired pending query");
ctx->tg->send_query(td_api::make_object<td_api::editInlineMessageText>(
query.inline_message_id_,
nullptr, // reply_markup
static_cast<td_api::object_ptr<td_api::InputMessageContent>>(td_api::make_object<td_api::inputMessageText>(
td_api::make_object<td_api::formattedText>(
"<сообщение устарело>",
std::move(std::vector<td_api::object_ptr<td_api::textEntity>>())
),
nullptr,
false
))
), {});
}
}
bool cmd::shorten_link(std::string link, context *ctx, std::function<void(std::string)> cb) {
spdlog::debug("creating CURL request");
CURL *req = curl_easy_init();
if (!req) {
return false;
}
char *escapedParam = curl_easy_escape(req, link.data(), link.size());
std::string url("https://slavasil.ru/create?url=");
url += escapedParam;
curl_free(escapedParam);
curl_easy_setopt(req, CURLOPT_URL, url.c_str());
curl_easy_setopt(req, CURLOPT_WRITEFUNCTION, curl_receive_cb);
curl_easy_setopt(req, CURLOPT_WRITEDATA, req);
curl_easy_setopt(req, CURLOPT_PRIVATE, ctx);
curl_easy_setopt(req, CURLOPT_FOLLOWLOCATION, 1);
ctx->requests.emplace(req, [cb, req, ctx](active_request &r){
std::string shortenedUrl(r.receivedData.data(), r.receivedData.size());
spdlog::info("Received data from HTTP server: {}", shortenedUrl);
cb(shortenedUrl);
ctx->requests.erase(req);
});
CURLMcode r = curl_multi_add_handle(ctx->curl, req);
return r == CURLM_OK;
}
uint64_t cmd::new_pending_query_id() {
static std::mt19937 rng;
static std::uniform_int_distribution<uint64_t> dist(0, 0xFFFFFFFFFFFFFFFFULL);
return dist(rng);
}