yalantinglibs/include/ylt/coro_rpc/impl/context.hpp

193 lines
5.1 KiB
C++

/*
* Copyright (c) 2023, Alibaba Group Holding Limited;
*
* 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 <async_simple/coro/Lazy.h>
#include <any>
#include <atomic>
#include <cstdint>
#include <memory>
#include <type_traits>
#include <utility>
#include <variant>
#include <ylt/easylog.hpp>
#include "coro_connection.hpp"
#include "ylt/util/type_traits.h"
namespace coro_rpc {
/*!
*
* @tparam return_msg_type
* @tparam coro_connection
*/
template <typename return_msg_type, typename rpc_protocol>
class context_base {
protected:
std::shared_ptr<context_info_t<rpc_protocol>> self_;
typename rpc_protocol::req_header &get_req_head() { return self_->req_head_; }
public:
/*!
* Construct a context by a share pointer of context Concept
* instance
* @param a share pointer for coro_connection
*/
context_base(std::shared_ptr<context_info_t<rpc_protocol>> context_info)
: self_(std::move(context_info)) {
if (self_->conn_) {
self_->conn_->set_rpc_call_type(
coro_connection::rpc_call_type::callback_started);
}
};
context_base() = default;
using return_type = return_msg_type;
/*!
* Send response message
*
* The `return_msg_type` must be constructed by these `args`.
*
* If the connection has already sent response message,
* an error log will be reported.
*
* @tparam Args
* @param args
*/
template <typename... Args>
void response_msg(Args &&...args) {
if constexpr (std::is_same_v<return_msg_type, void>) {
static_assert(sizeof...(args) == 0, "illegal args");
auto old_flag = self_->has_response_.exchange(true);
if (old_flag != false)
AS_UNLIKELY {
ELOGV(ERROR, "response message more than one time");
return;
}
if (has_closed())
AS_UNLIKELY {
ELOGV(DEBUG, "response_msg failed: connection has been closed");
return;
}
std::visit(
[&]<typename serialize_proto>(const serialize_proto &) {
self_->conn_->template response_msg<rpc_protocol>(
serialize_proto::serialize(), self_->req_head_,
self_->is_delay_);
},
*rpc_protocol::get_serialize_protocol(self_->req_head_));
}
else {
static_assert(
requires { return_msg_type{std::forward<Args>(args)...}; },
"constructed return_msg_type failed by illegal args");
return_msg_type ret{std::forward<Args>(args)...};
auto old_flag = self_->has_response_.exchange(true);
if (old_flag != false)
AS_UNLIKELY {
ELOGV(ERROR, "response message more than one time");
return;
}
if (has_closed())
AS_UNLIKELY {
ELOGV(DEBUG, "response_msg failed: connection has been closed");
return;
}
std::visit(
[&]<typename serialize_proto>(const serialize_proto &) {
self_->conn_->template response_msg<rpc_protocol>(
serialize_proto::serialize(ret), self_->req_head_,
self_->is_delay_);
},
*rpc_protocol::get_serialize_protocol(self_->req_head_));
// response_handler_(std::move(conn_), std::move(ret));
}
}
/*!
* Check connection closed or not
*
* @return true if closed, otherwise false
*/
bool has_closed() const { return self_->conn_->has_closed(); }
void set_delay() {
self_->is_delay_ = true;
self_->conn_->set_rpc_call_type(
coro_connection::rpc_call_type::callback_with_delay);
}
template <typename T>
void set_tag(T &&tag) {
self_->conn_->set_tag(std::forward<T>(tag));
}
std::any get_tag() { return self_->conn_->get_tag(); }
};
template <typename T>
struct get_type_t {
using type = T;
};
template <typename T>
struct get_type_t<async_simple::coro::Lazy<T>> {
using type = T;
};
template <auto func>
inline auto get_return_type() {
using T = decltype(func);
using param_type = function_parameters_t<T>;
using return_type = typename get_type_t<function_return_type_t<T>>::type;
if constexpr (std::is_void_v<param_type>) {
if constexpr (std::is_void_v<return_type>) {
return;
}
else {
return return_type{};
}
}
else {
using First = std::tuple_element_t<0, param_type>;
constexpr bool is_conn = is_specialization<First, context_base>::value;
if constexpr (is_conn) {
using U = typename First::return_type;
if constexpr (std::is_void_v<U>) {
return;
}
else {
return U{};
}
}
else if constexpr (!std::is_same_v<void, return_type>) {
return return_type{};
}
else {
return;
}
}
}
} // namespace coro_rpc