discord-bot/libs/DPP-10.0.29/include/dpp/sync.h

82 lines
3.1 KiB
C++

/************************************************************************************
*
* D++, A Lightweight C++ library for Discord
*
* Copyright 2022 Craig Edwards and D++ contributors
* (https://github.com/brainboxdotcc/DPP/graphs/contributors)
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
************************************************************************************/
#pragma once
#include <dpp/export.h>
#include <dpp/snowflake.h>
#include <future>
#include <utility>
#include <dpp/exception.h>
namespace dpp {
/**
* @brief Call a D++ REST function synchronously.
*
* Synchronously calling a REST function means *IT WILL BLOCK* - This is a Bad Thing™ and strongly discouraged.
* There are very few circumstances you actually need this. If you do need to use this, you'll know it.
*
* Example:
*
* ```cpp
* dpp::message m = dpp::sync<dpp::message>(&bot, &dpp::cluster::message_create, dpp::message(channel_id, "moo."));
* ```
*
* @warning As previously mentioned, this template will block. It is ill-advised to call this outside of
* a separate thread and this should never be directly used in any event such as dpp::cluster::on_interaction_create!
* @tparam T type of expected return value, should match up with the method called
* @tparam F Type of class method in dpp::cluster to call.
* @tparam Ts Function parameters in method call
* @param c A pointer to dpp::cluster object
* @param func pointer to class method in dpp::cluster to call. This can call any
* dpp::cluster member function who's last parameter is a dpp::command_completion_event_t callback type.
* @param args Zero or more arguments for the method call
* @return An instantiated object of type T
* @throw dpp::rest_exception On failure of the method call, an exception is thrown
*/
template<typename T, class F, class... Ts> T sync(class cluster* c, F func, Ts&&... args) {
std::promise<T> _p;
std::future<T> _f = _p.get_future();
/* (obj ->* func) is the obscure syntax for calling a method pointer on an object instance */
(c ->* func)(std::forward<Ts>(args)..., [&_p](const auto& cc) {
try {
if (cc.is_error()) {
const auto& error = cc.get_error();
throw dpp::rest_exception((exception_error_code)error.code, error.message);
} else {
try {
_p.set_value(std::get<T>(cc.value));
} catch (const std::exception& e) {
throw dpp::rest_exception(err_unknown, e.what());
}
}
} catch (const dpp::rest_exception&) {
_p.set_exception(std::current_exception());
}
});
/* Blocking calling thread until rest request is completed.
* Exceptions encountered on the other thread are re-thrown.
*/
return _f.get();
}
} // namespace dpp