2018-12-31 22:04:05 +03:00
|
|
|
//
|
2018-01-02 16:42:31 +03:00
|
|
|
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018
|
2018-12-31 22:04:05 +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/ClientJson.h"
|
|
|
|
|
|
|
|
#include "td/telegram/td_api.h"
|
|
|
|
#include "td/telegram/td_api_json.h"
|
|
|
|
|
|
|
|
#include "td/tl/tl_json.h"
|
|
|
|
|
|
|
|
#include "td/utils/format.h"
|
|
|
|
#include "td/utils/JsonBuilder.h"
|
|
|
|
#include "td/utils/logging.h"
|
|
|
|
#include "td/utils/Status.h"
|
|
|
|
|
|
|
|
namespace td {
|
|
|
|
|
|
|
|
Result<Client::Request> ClientJson::to_request(Slice request) {
|
|
|
|
auto request_str = request.str();
|
|
|
|
TRY_RESULT(json_value, json_decode(request_str));
|
|
|
|
if (json_value.type() != JsonValue::Type::Object) {
|
2018-05-20 15:30:36 +03:00
|
|
|
return Status::Error("Expected an Object");
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
std::uint64_t extra_id = extra_id_.fetch_add(1, std::memory_order_relaxed);
|
2018-05-20 16:16:16 +03:00
|
|
|
if (has_json_object_field(json_value.get_object(), "@extra")) {
|
2018-12-31 22:04:05 +03:00
|
|
|
std::lock_guard<std::mutex> guard(mutex_);
|
2018-05-20 16:16:16 +03:00
|
|
|
extra_[extra_id] = json_encode<string>(
|
|
|
|
get_json_object_field(json_value.get_object(), "@extra", JsonValue::Type::Null).move_as_ok());
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
td_api::object_ptr<td_api::Function> func;
|
|
|
|
TRY_STATUS(from_json(func, json_value));
|
|
|
|
return Client::Request{extra_id, std::move(func)};
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string ClientJson::from_response(Client::Response response) {
|
|
|
|
auto str = json_encode<string>(ToJson(static_cast<td_api::Object &>(*response.object)));
|
|
|
|
CHECK(!str.empty() && str.back() == '}');
|
|
|
|
std::string extra;
|
|
|
|
if (response.id != 0) {
|
|
|
|
std::lock_guard<std::mutex> guard(mutex_);
|
|
|
|
auto it = extra_.find(response.id);
|
|
|
|
if (it != extra_.end()) {
|
|
|
|
extra = std::move(it->second);
|
|
|
|
extra_.erase(it);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!extra.empty()) {
|
|
|
|
str.pop_back();
|
|
|
|
str.reserve(str.size() + 10 + extra.size());
|
|
|
|
str += ",\"@extra\":";
|
|
|
|
str += extra;
|
|
|
|
str += "}";
|
|
|
|
}
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
TD_THREAD_LOCAL std::string *ClientJson::current_output_;
|
|
|
|
CSlice ClientJson::store_string(std::string str) {
|
|
|
|
init_thread_local<std::string>(ClientJson::current_output_);
|
|
|
|
*current_output_ = std::move(str);
|
|
|
|
return *current_output_;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ClientJson::send(Slice request) {
|
2018-05-20 02:56:41 +03:00
|
|
|
auto r_request = to_request(request);
|
|
|
|
if (r_request.is_error()) {
|
|
|
|
LOG(ERROR) << "Failed to parse " << tag("request", format::escaped(request)) << " " << r_request.error();
|
|
|
|
return;
|
|
|
|
}
|
2018-12-31 22:04:05 +03:00
|
|
|
|
2018-05-20 02:56:41 +03:00
|
|
|
client_.send(r_request.move_as_ok());
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
CSlice ClientJson::receive(double timeout) {
|
|
|
|
auto response = client_.receive(timeout);
|
|
|
|
if (!response.object) {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
return store_string(from_response(std::move(response)));
|
|
|
|
}
|
|
|
|
|
|
|
|
CSlice ClientJson::execute(Slice request) {
|
|
|
|
auto r_request = to_request(request);
|
|
|
|
if (r_request.is_error()) {
|
|
|
|
LOG(ERROR) << "Failed to parse " << tag("request", format::escaped(request)) << " " << r_request.error();
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
return store_string(from_response(Client::execute(r_request.move_as_ok())));
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace td
|