2021-06-21 02:08:11 +03:00
|
|
|
//
|
2024-01-01 03:07:21 +03:00
|
|
|
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2024
|
2021-06-21 02:08:11 +03:00
|
|
|
//
|
|
|
|
// Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
|
|
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
|
|
//
|
|
|
|
#include "td/telegram/BotCommand.h"
|
|
|
|
|
2021-06-22 04:25:15 +03:00
|
|
|
#include "td/telegram/BotCommandScope.h"
|
2021-06-21 02:08:11 +03:00
|
|
|
#include "td/telegram/ContactsManager.h"
|
2021-06-23 03:22:23 +03:00
|
|
|
#include "td/telegram/Global.h"
|
2021-06-22 04:25:15 +03:00
|
|
|
#include "td/telegram/misc.h"
|
2021-06-21 02:08:11 +03:00
|
|
|
#include "td/telegram/Td.h"
|
2023-07-01 14:53:04 +03:00
|
|
|
#include "td/telegram/telegram_api.h"
|
2021-06-21 02:08:11 +03:00
|
|
|
|
|
|
|
#include "td/utils/algorithm.h"
|
2021-06-22 04:25:15 +03:00
|
|
|
#include "td/utils/buffer.h"
|
|
|
|
#include "td/utils/logging.h"
|
2021-06-23 03:22:23 +03:00
|
|
|
#include "td/utils/misc.h"
|
|
|
|
#include "td/utils/SliceBuilder.h"
|
2021-06-22 04:25:15 +03:00
|
|
|
#include "td/utils/Status.h"
|
2021-06-23 03:22:23 +03:00
|
|
|
#include "td/utils/utf8.h"
|
2021-06-21 02:08:11 +03:00
|
|
|
|
2024-01-08 15:42:21 +03:00
|
|
|
#include <algorithm>
|
|
|
|
|
2021-06-21 02:08:11 +03:00
|
|
|
namespace td {
|
|
|
|
|
2021-07-04 05:58:54 +03:00
|
|
|
class SetBotCommandsQuery final : public Td::ResultHandler {
|
2021-06-22 04:25:15 +03:00
|
|
|
Promise<Unit> promise_;
|
|
|
|
|
|
|
|
public:
|
|
|
|
explicit SetBotCommandsQuery(Promise<Unit> &&promise) : promise_(std::move(promise)) {
|
|
|
|
}
|
|
|
|
|
2021-06-22 04:34:29 +03:00
|
|
|
void send(BotCommandScope scope, const string &language_code, vector<BotCommand> &&commands) {
|
2021-06-22 04:25:15 +03:00
|
|
|
send_query(G()->net_query_creator().create(telegram_api::bots_setBotCommands(
|
2021-11-08 15:20:38 +03:00
|
|
|
scope.get_input_bot_command_scope(td_), language_code,
|
2021-06-22 04:34:29 +03:00
|
|
|
transform(commands, [](const BotCommand &command) { return command.get_input_bot_command(); }))));
|
2021-06-22 04:25:15 +03:00
|
|
|
}
|
|
|
|
|
2021-11-08 14:19:57 +03:00
|
|
|
void on_result(BufferSlice packet) final {
|
2021-06-22 04:25:15 +03:00
|
|
|
auto result_ptr = fetch_result<telegram_api::bots_setBotCommands>(packet);
|
|
|
|
if (result_ptr.is_error()) {
|
2021-11-08 14:19:57 +03:00
|
|
|
return on_error(result_ptr.move_as_error());
|
2021-06-22 04:25:15 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!result_ptr.ok()) {
|
|
|
|
LOG(ERROR) << "Set bot commands request failed";
|
|
|
|
}
|
|
|
|
promise_.set_value(Unit());
|
|
|
|
}
|
|
|
|
|
2021-11-08 14:19:57 +03:00
|
|
|
void on_error(Status status) final {
|
2021-06-22 04:25:15 +03:00
|
|
|
promise_.set_error(std::move(status));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-07-04 05:58:54 +03:00
|
|
|
class ResetBotCommandsQuery final : public Td::ResultHandler {
|
2021-06-22 04:25:15 +03:00
|
|
|
Promise<Unit> promise_;
|
|
|
|
|
|
|
|
public:
|
|
|
|
explicit ResetBotCommandsQuery(Promise<Unit> &&promise) : promise_(std::move(promise)) {
|
|
|
|
}
|
|
|
|
|
|
|
|
void send(BotCommandScope scope, const string &language_code) {
|
|
|
|
send_query(G()->net_query_creator().create(
|
2021-11-08 15:20:38 +03:00
|
|
|
telegram_api::bots_resetBotCommands(scope.get_input_bot_command_scope(td_), language_code)));
|
2021-06-22 04:25:15 +03:00
|
|
|
}
|
|
|
|
|
2021-11-08 14:19:57 +03:00
|
|
|
void on_result(BufferSlice packet) final {
|
2021-06-22 04:25:15 +03:00
|
|
|
auto result_ptr = fetch_result<telegram_api::bots_resetBotCommands>(packet);
|
|
|
|
if (result_ptr.is_error()) {
|
2021-11-08 14:19:57 +03:00
|
|
|
return on_error(result_ptr.move_as_error());
|
2021-06-22 04:25:15 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
promise_.set_value(Unit());
|
|
|
|
}
|
|
|
|
|
2021-11-08 14:19:57 +03:00
|
|
|
void on_error(Status status) final {
|
2021-06-22 04:25:15 +03:00
|
|
|
promise_.set_error(std::move(status));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-07-04 05:58:54 +03:00
|
|
|
class GetBotCommandsQuery final : public Td::ResultHandler {
|
2021-06-22 04:25:15 +03:00
|
|
|
Promise<td_api::object_ptr<td_api::botCommands>> promise_;
|
|
|
|
|
|
|
|
public:
|
|
|
|
explicit GetBotCommandsQuery(Promise<td_api::object_ptr<td_api::botCommands>> &&promise)
|
|
|
|
: promise_(std::move(promise)) {
|
|
|
|
}
|
|
|
|
|
|
|
|
void send(BotCommandScope scope, const string &language_code) {
|
|
|
|
send_query(G()->net_query_creator().create(
|
2021-11-08 15:20:38 +03:00
|
|
|
telegram_api::bots_getBotCommands(scope.get_input_bot_command_scope(td_), language_code)));
|
2021-06-22 04:25:15 +03:00
|
|
|
}
|
|
|
|
|
2021-11-08 14:19:57 +03:00
|
|
|
void on_result(BufferSlice packet) final {
|
2021-06-22 04:25:15 +03:00
|
|
|
auto result_ptr = fetch_result<telegram_api::bots_getBotCommands>(packet);
|
|
|
|
if (result_ptr.is_error()) {
|
2021-11-08 14:19:57 +03:00
|
|
|
return on_error(result_ptr.move_as_error());
|
2021-06-22 04:25:15 +03:00
|
|
|
}
|
|
|
|
|
2021-11-08 15:20:38 +03:00
|
|
|
BotCommands commands(td_->contacts_manager_->get_my_id(), result_ptr.move_as_ok());
|
|
|
|
promise_.set_value(commands.get_bot_commands_object(td_));
|
2021-06-22 04:25:15 +03:00
|
|
|
}
|
|
|
|
|
2021-11-08 14:19:57 +03:00
|
|
|
void on_error(Status status) final {
|
2021-06-22 04:25:15 +03:00
|
|
|
promise_.set_error(std::move(status));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-06-21 02:08:11 +03:00
|
|
|
BotCommand::BotCommand(telegram_api::object_ptr<telegram_api::botCommand> &&bot_command) {
|
|
|
|
CHECK(bot_command != nullptr);
|
|
|
|
command_ = std::move(bot_command->command_);
|
|
|
|
description_ = std::move(bot_command->description_);
|
|
|
|
}
|
|
|
|
|
|
|
|
td_api::object_ptr<td_api::botCommand> BotCommand::get_bot_command_object() const {
|
|
|
|
return td_api::make_object<td_api::botCommand>(command_, description_);
|
|
|
|
}
|
|
|
|
|
2021-06-22 04:34:29 +03:00
|
|
|
telegram_api::object_ptr<telegram_api::botCommand> BotCommand::get_input_bot_command() const {
|
|
|
|
return telegram_api::make_object<telegram_api::botCommand>(command_, description_);
|
|
|
|
}
|
|
|
|
|
2021-06-21 02:36:04 +03:00
|
|
|
bool operator==(const BotCommand &lhs, const BotCommand &rhs) {
|
|
|
|
return lhs.command_ == rhs.command_ && lhs.description_ == rhs.description_;
|
|
|
|
}
|
|
|
|
|
2021-06-21 02:08:11 +03:00
|
|
|
BotCommands::BotCommands(UserId bot_user_id, vector<telegram_api::object_ptr<telegram_api::botCommand>> &&bot_commands)
|
|
|
|
: bot_user_id_(bot_user_id) {
|
|
|
|
commands_ = transform(std::move(bot_commands), [](telegram_api::object_ptr<telegram_api::botCommand> &&bot_command) {
|
|
|
|
return BotCommand(std::move(bot_command));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
td_api::object_ptr<td_api::botCommands> BotCommands::get_bot_commands_object(Td *td) const {
|
|
|
|
auto commands = transform(commands_, [](const auto &command) { return command.get_bot_command_object(); });
|
|
|
|
return td_api::make_object<td_api::botCommands>(
|
|
|
|
td->contacts_manager_->get_user_id_object(bot_user_id_, "get_bot_commands_object"), std::move(commands));
|
|
|
|
}
|
|
|
|
|
2024-01-08 01:44:58 +03:00
|
|
|
bool BotCommands::update_all_bot_commands(vector<BotCommands> &all_bot_commands, BotCommands &&bot_commands) {
|
|
|
|
auto is_from_bot = [bot_user_id = bot_commands.bot_user_id_](const BotCommands &commands) {
|
|
|
|
return commands.bot_user_id_ == bot_user_id;
|
|
|
|
};
|
|
|
|
|
|
|
|
if (bot_commands.commands_.empty()) {
|
|
|
|
return td::remove_if(all_bot_commands, is_from_bot);
|
|
|
|
}
|
|
|
|
auto it = std::find_if(all_bot_commands.begin(), all_bot_commands.end(), is_from_bot);
|
|
|
|
if (it != all_bot_commands.end()) {
|
|
|
|
if (*it != bot_commands) {
|
|
|
|
*it = std::move(bot_commands);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
all_bot_commands.push_back(std::move(bot_commands));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-06-21 02:36:04 +03:00
|
|
|
bool operator==(const BotCommands &lhs, const BotCommands &rhs) {
|
|
|
|
return lhs.bot_user_id_ == rhs.bot_user_id_ && lhs.commands_ == rhs.commands_;
|
|
|
|
}
|
|
|
|
|
2021-06-22 04:25:15 +03:00
|
|
|
void set_commands(Td *td, td_api::object_ptr<td_api::BotCommandScope> &&scope_ptr, string &&language_code,
|
|
|
|
vector<td_api::object_ptr<td_api::botCommand>> &&commands, Promise<Unit> &&promise) {
|
|
|
|
TRY_RESULT_PROMISE(promise, scope, BotCommandScope::get_bot_command_scope(td, std::move(scope_ptr)));
|
2023-02-27 01:17:22 +03:00
|
|
|
TRY_STATUS_PROMISE(promise, validate_bot_language_code(language_code));
|
2021-06-22 04:25:15 +03:00
|
|
|
|
2021-06-22 04:34:29 +03:00
|
|
|
vector<BotCommand> new_commands;
|
2021-06-22 04:25:15 +03:00
|
|
|
for (auto &command : commands) {
|
|
|
|
if (command == nullptr) {
|
|
|
|
return promise.set_error(Status::Error(400, "Command must be non-empty"));
|
|
|
|
}
|
|
|
|
if (!clean_input_string(command->command_)) {
|
|
|
|
return promise.set_error(Status::Error(400, "Command must be encoded in UTF-8"));
|
|
|
|
}
|
|
|
|
if (!clean_input_string(command->description_)) {
|
|
|
|
return promise.set_error(Status::Error(400, "Command description must be encoded in UTF-8"));
|
|
|
|
}
|
|
|
|
|
|
|
|
const size_t MAX_COMMAND_TEXT_LENGTH = 32;
|
|
|
|
command->command_ = trim(command->command_);
|
|
|
|
if (command->command_[0] == '/') {
|
|
|
|
command->command_ = command->command_.substr(1);
|
|
|
|
}
|
|
|
|
if (command->command_.empty()) {
|
|
|
|
return promise.set_error(Status::Error(400, "Command must be non-empty"));
|
|
|
|
}
|
|
|
|
if (utf8_length(command->command_) > MAX_COMMAND_TEXT_LENGTH) {
|
|
|
|
return promise.set_error(
|
|
|
|
Status::Error(400, PSLICE() << "Command length must not exceed " << MAX_COMMAND_TEXT_LENGTH));
|
|
|
|
}
|
|
|
|
|
|
|
|
const size_t MAX_COMMAND_DESCRIPTION_LENGTH = 256;
|
|
|
|
command->description_ = trim(command->description_);
|
|
|
|
auto description_length = utf8_length(command->description_);
|
2021-07-24 06:00:31 +03:00
|
|
|
if (command->description_.empty()) {
|
|
|
|
return promise.set_error(Status::Error(400, "Command description must be non-empty"));
|
2021-06-22 04:25:15 +03:00
|
|
|
}
|
|
|
|
if (description_length > MAX_COMMAND_DESCRIPTION_LENGTH) {
|
|
|
|
return promise.set_error(Status::Error(
|
|
|
|
400, PSLICE() << "Command description length must not exceed " << MAX_COMMAND_DESCRIPTION_LENGTH));
|
|
|
|
}
|
|
|
|
|
|
|
|
new_commands.emplace_back(std::move(command->command_), std::move(command->description_));
|
|
|
|
}
|
|
|
|
|
|
|
|
td->create_handler<SetBotCommandsQuery>(std::move(promise))->send(scope, language_code, std::move(new_commands));
|
|
|
|
}
|
|
|
|
|
|
|
|
void delete_commands(Td *td, td_api::object_ptr<td_api::BotCommandScope> &&scope_ptr, string &&language_code,
|
|
|
|
Promise<Unit> &&promise) {
|
|
|
|
TRY_RESULT_PROMISE(promise, scope, BotCommandScope::get_bot_command_scope(td, std::move(scope_ptr)));
|
2023-02-27 01:17:22 +03:00
|
|
|
TRY_STATUS_PROMISE(promise, validate_bot_language_code(language_code));
|
2021-06-22 04:25:15 +03:00
|
|
|
|
|
|
|
td->create_handler<ResetBotCommandsQuery>(std::move(promise))->send(scope, language_code);
|
|
|
|
}
|
|
|
|
|
|
|
|
void get_commands(Td *td, td_api::object_ptr<td_api::BotCommandScope> &&scope_ptr, string &&language_code,
|
|
|
|
Promise<td_api::object_ptr<td_api::botCommands>> &&promise) {
|
|
|
|
TRY_RESULT_PROMISE(promise, scope, BotCommandScope::get_bot_command_scope(td, std::move(scope_ptr)));
|
2023-02-27 01:17:22 +03:00
|
|
|
TRY_STATUS_PROMISE(promise, validate_bot_language_code(language_code));
|
2021-06-22 04:25:15 +03:00
|
|
|
|
|
|
|
td->create_handler<GetBotCommandsQuery>(std::move(promise))->send(scope, language_code);
|
|
|
|
}
|
|
|
|
|
2021-06-21 02:08:11 +03:00
|
|
|
} // namespace td
|