mirror of
https://github.com/ton-blockchain/ton
synced 2025-02-12 11:12:16 +00:00
* Bugfixes in rldp-http-proxy and http parser * Tonlib: change liteservers on query timeout or connection close * Increase maximum size of http request * Minor bugfixes in http
110 lines
3.3 KiB
C++
110 lines
3.3 KiB
C++
/*
|
|
This file is part of TON Blockchain Library.
|
|
|
|
TON Blockchain Library is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU Lesser General Public License as published by
|
|
the Free Software Foundation, either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
TON Blockchain Library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public License
|
|
along with TON Blockchain Library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
Copyright 2019-2020 Telegram Systems LLP
|
|
*/
|
|
#include "http-outbound-connection.h"
|
|
|
|
#include "td/utils/port/StdStreams.h"
|
|
|
|
namespace ton {
|
|
|
|
namespace http {
|
|
|
|
td::Status HttpOutboundConnection::receive(td::ChainBufferReader &input) {
|
|
if (input.size() == 0) {
|
|
return td::Status::OK();
|
|
}
|
|
if (reading_payload_) {
|
|
return receive_payload(input);
|
|
}
|
|
if (!promise_) {
|
|
return td::Status::Error("unexpected data");
|
|
}
|
|
|
|
while (!cur_response_ || !cur_response_->check_parse_header_completed()) {
|
|
bool exit_loop;
|
|
auto R = HttpResponse::parse(std::move(cur_response_), cur_line_, force_no_payload_, keep_alive_, exit_loop, input);
|
|
if (R.is_error()) {
|
|
answer_error(HttpStatusCode::status_bad_request, "", std::move(promise_));
|
|
return td::Status::OK();
|
|
}
|
|
cur_response_ = R.move_as_ok();
|
|
if (exit_loop) {
|
|
return td::Status::OK();
|
|
}
|
|
}
|
|
|
|
if (cur_response_->code() == 100) {
|
|
cur_response_ = nullptr;
|
|
return td::Status::OK();
|
|
}
|
|
|
|
close_after_read_ = !cur_response_->keep_alive() || !keep_alive_;
|
|
|
|
auto payload = cur_response_->create_empty_payload().move_as_ok();
|
|
promise_.set_value(std::make_pair(std::move(cur_response_), payload));
|
|
read_payload(std::move(payload));
|
|
|
|
if (!reading_payload_) {
|
|
return td::Status::OK();
|
|
}
|
|
return receive_payload(input);
|
|
}
|
|
|
|
void HttpOutboundConnection::send_query(
|
|
std::unique_ptr<HttpRequest> request, std::shared_ptr<HttpPayload> payload, td::Timestamp timeout,
|
|
td::Promise<std::pair<std::unique_ptr<HttpResponse>, std::shared_ptr<HttpPayload>>> promise) {
|
|
CHECK(request);
|
|
CHECK(payload);
|
|
if (promise_) {
|
|
LOG(INFO) << "delaying send of HTTP request";
|
|
next_.push_back(Query{std::move(request), std::move(payload), timeout, std::move(promise)});
|
|
return;
|
|
}
|
|
LOG(INFO) << "sending HTTP request";
|
|
keep_alive_ = request->keep_alive();
|
|
force_no_payload_ = request->no_payload_in_answer();
|
|
request->store_http(buffered_fd_.output_buffer());
|
|
write_payload(std::move(payload));
|
|
promise_ = std::move(promise);
|
|
alarm_timestamp() = timeout;
|
|
|
|
loop();
|
|
}
|
|
|
|
void HttpOutboundConnection::send_next_query() {
|
|
if (next_.size() == 0) {
|
|
return;
|
|
}
|
|
|
|
LOG(INFO) << "sending delayed HTTP request";
|
|
auto p = std::move(next_.front());
|
|
next_.pop_front();
|
|
keep_alive_ = p.request->keep_alive();
|
|
force_no_payload_ = p.request->no_payload_in_answer();
|
|
|
|
p.request->store_http(buffered_fd_.output_buffer());
|
|
write_payload(std::move(p.payload));
|
|
alarm_timestamp() = p.timeout;
|
|
promise_ = std::move(p.promise);
|
|
|
|
loop();
|
|
}
|
|
|
|
} // namespace http
|
|
|
|
} // namespace ton
|