469 lines
16 KiB
C++
469 lines
16 KiB
C++
/*
|
|
|
|
Copyright (c) 2008-2018, Arvid Norberg
|
|
All rights reserved.
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
modification, are permitted provided that the following conditions
|
|
are met:
|
|
|
|
* Redistributions of source code must retain the above copyright
|
|
notice, this list of conditions and the following disclaimer.
|
|
* Redistributions in binary form must reproduce the above copyright
|
|
notice, this list of conditions and the following disclaimer in
|
|
the documentation and/or other materials provided with the distribution.
|
|
* Neither the name of the author nor the names of its
|
|
contributors may be used to endorse or promote products derived
|
|
from this software without specific prior written permission.
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
#include <cinttypes> // for PRId64 et.al.
|
|
|
|
#include "libtorrent/config.hpp"
|
|
#include "libtorrent/http_seed_connection.hpp"
|
|
#include "libtorrent/invariant_check.hpp"
|
|
#include "libtorrent/aux_/session_impl.hpp"
|
|
#include "libtorrent/peer_info.hpp"
|
|
#include "libtorrent/hex.hpp" // for is_hex
|
|
#include "libtorrent/optional.hpp"
|
|
|
|
namespace libtorrent {
|
|
|
|
http_seed_connection::http_seed_connection(peer_connection_args const& pack
|
|
, web_seed_t& web)
|
|
: web_connection_base(pack, web)
|
|
, m_url(web.url)
|
|
, m_web(&web)
|
|
, m_response_left(0)
|
|
, m_chunk_pos(0)
|
|
, m_partial_chunk_header(0)
|
|
{
|
|
INVARIANT_CHECK;
|
|
|
|
if (!m_settings.get_bool(settings_pack::report_web_seed_downloads))
|
|
ignore_stats(true);
|
|
|
|
std::shared_ptr<torrent> tor = pack.tor.lock();
|
|
TORRENT_ASSERT(tor);
|
|
int blocks_per_piece = tor->torrent_file().piece_length() / tor->block_size();
|
|
|
|
// multiply with the blocks per piece since that many requests are
|
|
// merged into one http request
|
|
max_out_request_queue(m_settings.get_int(settings_pack::urlseed_pipeline_size)
|
|
* blocks_per_piece);
|
|
|
|
prefer_contiguous_blocks(blocks_per_piece);
|
|
|
|
#ifndef TORRENT_DISABLE_LOGGING
|
|
peer_log(peer_log_alert::info, "CONNECT", "http_seed_connection");
|
|
#endif
|
|
}
|
|
|
|
void http_seed_connection::on_connected()
|
|
{
|
|
// this is always a seed
|
|
incoming_have_all();
|
|
web_connection_base::on_connected();
|
|
}
|
|
|
|
void http_seed_connection::disconnect(error_code const& ec
|
|
, operation_t const op, disconnect_severity_t const error)
|
|
{
|
|
if (is_disconnecting()) return;
|
|
|
|
if (op == operation_t::connect && m_web && !m_web->endpoints.empty())
|
|
{
|
|
// we failed to connect to this IP. remove it so that the next attempt
|
|
// uses the next IP in the list.
|
|
m_web->endpoints.erase(m_web->endpoints.begin());
|
|
}
|
|
|
|
std::shared_ptr<torrent> t = associated_torrent().lock();
|
|
peer_connection::disconnect(ec, op, error);
|
|
if (t) t->disconnect_web_seed(this);
|
|
}
|
|
|
|
piece_block_progress http_seed_connection::downloading_piece_progress() const
|
|
{
|
|
if (m_requests.empty()) return {};
|
|
|
|
std::shared_ptr<torrent> t = associated_torrent().lock();
|
|
TORRENT_ASSERT(t);
|
|
|
|
piece_block_progress ret;
|
|
|
|
peer_request const& pr = m_requests.front();
|
|
ret.piece_index = pr.piece;
|
|
if (!m_parser.header_finished())
|
|
{
|
|
ret.bytes_downloaded = 0;
|
|
}
|
|
else
|
|
{
|
|
int const receive_buffer_size = int(m_recv_buffer.get().size()) - m_parser.body_start();
|
|
// this is an approximation. in chunked encoding mode the chunk headers
|
|
// should really be subtracted from the receive_buffer_size
|
|
ret.bytes_downloaded = std::max(0, t->block_size() - receive_buffer_size);
|
|
}
|
|
// this is used to make sure that the block_index stays within
|
|
// bounds. If the entire piece is downloaded, the block_index
|
|
// would otherwise point to one past the end
|
|
int const correction = ret.bytes_downloaded ? -1 : 0;
|
|
ret.block_index = (pr.start + ret.bytes_downloaded + correction) / t->block_size();
|
|
ret.full_block_bytes = t->block_size();
|
|
piece_index_t const last_piece = t->torrent_file().last_piece();
|
|
if (ret.piece_index == last_piece && ret.block_index
|
|
== t->torrent_file().piece_size(last_piece) / t->block_size())
|
|
ret.full_block_bytes = t->torrent_file().piece_size(last_piece) % t->block_size();
|
|
return ret;
|
|
}
|
|
|
|
void http_seed_connection::write_request(peer_request const& r)
|
|
{
|
|
INVARIANT_CHECK;
|
|
|
|
std::shared_ptr<torrent> t = associated_torrent().lock();
|
|
TORRENT_ASSERT(t);
|
|
|
|
TORRENT_ASSERT(t->valid_metadata());
|
|
// http_seeds don't support requesting more than one piece
|
|
// at a time
|
|
TORRENT_ASSERT(r.length <= t->torrent_file().piece_size(r.piece));
|
|
|
|
std::string request;
|
|
request.reserve(400);
|
|
|
|
int size = r.length;
|
|
const int bs = t->block_size();
|
|
const int piece_size = t->torrent_file().piece_length();
|
|
peer_request pr;
|
|
while (size > 0)
|
|
{
|
|
int request_offset = r.start + r.length - size;
|
|
pr.start = request_offset % piece_size;
|
|
pr.length = std::min(bs, size);
|
|
pr.piece = piece_index_t(static_cast<int>(r.piece) + request_offset / piece_size);
|
|
m_requests.push_back(pr);
|
|
size -= pr.length;
|
|
}
|
|
|
|
int proxy_type = m_settings.get_int(settings_pack::proxy_type);
|
|
bool using_proxy = (proxy_type == settings_pack::http
|
|
|| proxy_type == settings_pack::http_pw) && !m_ssl;
|
|
|
|
request += "GET ";
|
|
request += using_proxy ? m_url : m_path;
|
|
request += "?info_hash=";
|
|
request += escape_string({t->torrent_file().info_hash().data(), 20});
|
|
request += "&piece=";
|
|
request += to_string(r.piece);
|
|
|
|
// if we're requesting less than an entire piece we need to
|
|
// add ranges
|
|
if (r.start > 0 || r.length != t->torrent_file().piece_size(r.piece))
|
|
{
|
|
request += "&ranges=";
|
|
request += to_string(r.start).data();
|
|
request += "-";
|
|
// ranges are inclusive, just like HTTP
|
|
request += to_string(r.start + r.length - 1).data();
|
|
}
|
|
|
|
request += " HTTP/1.1\r\n";
|
|
add_headers(request, m_settings, using_proxy);
|
|
request += "\r\n\r\n";
|
|
m_first_request = false;
|
|
|
|
#ifndef TORRENT_DISABLE_LOGGING
|
|
peer_log(peer_log_alert::outgoing_message, "REQUEST", "%s", request.c_str());
|
|
#endif
|
|
|
|
send_buffer(request, message_type_request);
|
|
}
|
|
|
|
// --------------------------
|
|
// RECEIVE DATA
|
|
// --------------------------
|
|
|
|
void http_seed_connection::on_receive(error_code const& error
|
|
, std::size_t bytes_transferred)
|
|
{
|
|
INVARIANT_CHECK;
|
|
|
|
if (error)
|
|
{
|
|
received_bytes(0, int(bytes_transferred));
|
|
#ifndef TORRENT_DISABLE_LOGGING
|
|
if (should_log(peer_log_alert::info))
|
|
{
|
|
peer_log(peer_log_alert::info, "ERROR"
|
|
, "http_seed_connection error: %s", error.message().c_str());
|
|
}
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
std::shared_ptr<torrent> t = associated_torrent().lock();
|
|
TORRENT_ASSERT(t);
|
|
|
|
for (;;)
|
|
{
|
|
span<char const> recv_buffer = m_recv_buffer.get();
|
|
|
|
if (bytes_transferred == 0) break;
|
|
TORRENT_ASSERT(int(recv_buffer.size()) > 0);
|
|
|
|
TORRENT_ASSERT(!m_requests.empty());
|
|
if (m_requests.empty())
|
|
{
|
|
received_bytes(0, int(bytes_transferred));
|
|
disconnect(errors::http_error, operation_t::bittorrent, peer_error);
|
|
return;
|
|
}
|
|
|
|
peer_request front_request = m_requests.front();
|
|
|
|
bool header_finished = m_parser.header_finished();
|
|
if (!header_finished)
|
|
{
|
|
bool parse_error = false;
|
|
int protocol = 0;
|
|
int payload = 0;
|
|
std::tie(payload, protocol) = m_parser.incoming(
|
|
recv_buffer, parse_error);
|
|
received_bytes(0, protocol);
|
|
TORRENT_ASSERT(bytes_transferred >= aux::numeric_cast<std::size_t>(protocol));
|
|
bytes_transferred -= aux::numeric_cast<std::size_t>(protocol);
|
|
#if TORRENT_USE_ASSERTS
|
|
if (payload > front_request.length) payload = front_request.length;
|
|
#endif
|
|
|
|
if (parse_error)
|
|
{
|
|
received_bytes(0, int(bytes_transferred));
|
|
disconnect(errors::http_parse_error, operation_t::bittorrent, peer_error);
|
|
return;
|
|
}
|
|
|
|
TORRENT_ASSERT(int(recv_buffer.size()) == 0 || recv_buffer.front() == 'H');
|
|
|
|
TORRENT_ASSERT(int(recv_buffer.size()) <= m_recv_buffer.packet_size());
|
|
|
|
// this means the entire status line hasn't been received yet
|
|
if (m_parser.status_code() == -1)
|
|
{
|
|
TORRENT_ASSERT(payload == 0);
|
|
TORRENT_ASSERT(bytes_transferred == 0);
|
|
break;
|
|
}
|
|
|
|
// if the status code is not one of the accepted ones, abort
|
|
if (!is_ok_status(m_parser.status_code()))
|
|
{
|
|
auto const retry_time = value_or(m_parser.header_duration("retry-after")
|
|
, seconds32(m_settings.get_int(settings_pack::urlseed_wait_retry)));
|
|
|
|
// temporarily unavailable, retry later
|
|
t->retry_web_seed(this, retry_time);
|
|
|
|
if (t->alerts().should_post<url_seed_alert>())
|
|
{
|
|
std::string const error_msg = to_string(m_parser.status_code()).data()
|
|
+ (" " + m_parser.message());
|
|
t->alerts().emplace_alert<url_seed_alert>(t->get_handle(), url()
|
|
, error_msg);
|
|
}
|
|
received_bytes(0, int(bytes_transferred));
|
|
disconnect(error_code(m_parser.status_code(), http_category()), operation_t::bittorrent, failure);
|
|
return;
|
|
}
|
|
if (!m_parser.header_finished())
|
|
{
|
|
TORRENT_ASSERT(payload == 0);
|
|
TORRENT_ASSERT(bytes_transferred == 0);
|
|
break;
|
|
}
|
|
}
|
|
|
|
// we just completed reading the header
|
|
if (!header_finished)
|
|
{
|
|
if (is_redirect(m_parser.status_code()))
|
|
{
|
|
// this means we got a redirection request
|
|
// look for the location header
|
|
std::string location = m_parser.header("location");
|
|
received_bytes(0, int(bytes_transferred));
|
|
|
|
if (location.empty())
|
|
{
|
|
// we should not try this server again.
|
|
t->remove_web_seed_conn(this, errors::missing_location, operation_t::bittorrent, peer_error);
|
|
return;
|
|
}
|
|
|
|
// add the redirected url and remove the current one
|
|
t->add_web_seed(location, web_seed_entry::http_seed);
|
|
t->remove_web_seed_conn(this, errors::redirecting, operation_t::bittorrent, peer_error);
|
|
return;
|
|
}
|
|
|
|
std::string const& server_version = m_parser.header("server");
|
|
if (!server_version.empty())
|
|
{
|
|
m_server_string = "URL seed @ ";
|
|
m_server_string += m_host;
|
|
m_server_string += " (";
|
|
m_server_string += server_version;
|
|
m_server_string += ")";
|
|
}
|
|
|
|
m_response_left = atol(m_parser.header("content-length").c_str());
|
|
if (m_response_left == -1)
|
|
{
|
|
received_bytes(0, int(bytes_transferred));
|
|
// we should not try this server again.
|
|
t->remove_web_seed_conn(this, errors::no_content_length, operation_t::bittorrent, peer_error);
|
|
return;
|
|
}
|
|
if (m_response_left != front_request.length)
|
|
{
|
|
received_bytes(0, int(bytes_transferred));
|
|
// we should not try this server again.
|
|
t->remove_web_seed_conn(this, errors::invalid_range, operation_t::bittorrent, peer_error);
|
|
return;
|
|
}
|
|
m_body_start = m_parser.body_start();
|
|
}
|
|
|
|
recv_buffer = recv_buffer.subspan(m_body_start);
|
|
|
|
// =========================
|
|
// === CHUNKED ENCODING ===
|
|
// =========================
|
|
while (m_parser.chunked_encoding()
|
|
&& m_chunk_pos >= 0
|
|
&& m_chunk_pos < recv_buffer.size())
|
|
{
|
|
int header_size = 0;
|
|
std::int64_t chunk_size = 0;
|
|
span<char const> chunk_start = recv_buffer.subspan(aux::numeric_cast<std::ptrdiff_t>(m_chunk_pos));
|
|
TORRENT_ASSERT(chunk_start[0] == '\r'
|
|
|| aux::is_hex(chunk_start[0]));
|
|
bool ret = m_parser.parse_chunk_header(chunk_start, &chunk_size, &header_size);
|
|
if (!ret)
|
|
{
|
|
TORRENT_ASSERT(bytes_transferred >= aux::numeric_cast<std::size_t>(chunk_start.size() - m_partial_chunk_header));
|
|
bytes_transferred -= aux::numeric_cast<std::size_t>(chunk_start.size() - m_partial_chunk_header);
|
|
received_bytes(0, aux::numeric_cast<int>(chunk_start.size() - m_partial_chunk_header));
|
|
m_partial_chunk_header = aux::numeric_cast<int>(chunk_start.size());
|
|
if (bytes_transferred == 0) return;
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
#ifndef TORRENT_DISABLE_LOGGING
|
|
peer_log(peer_log_alert::info, "CHUNKED_ENCODING"
|
|
, "parsed chunk: %" PRId64 " header_size: %d"
|
|
, chunk_size, header_size);
|
|
#endif
|
|
TORRENT_ASSERT(bytes_transferred >= aux::numeric_cast<std::size_t>(header_size - m_partial_chunk_header));
|
|
bytes_transferred -= aux::numeric_cast<std::size_t>(header_size - m_partial_chunk_header);
|
|
|
|
received_bytes(0, header_size - m_partial_chunk_header);
|
|
m_partial_chunk_header = 0;
|
|
TORRENT_ASSERT(chunk_size != 0 || chunk_start.size() <= header_size || chunk_start[header_size] == 'H');
|
|
// cut out the chunk header from the receive buffer
|
|
TORRENT_ASSERT(m_chunk_pos + m_body_start < INT_MAX);
|
|
m_recv_buffer.cut(header_size, t->block_size() + 1024, aux::numeric_cast<int>(m_chunk_pos + m_body_start));
|
|
recv_buffer = m_recv_buffer.get();
|
|
recv_buffer = recv_buffer.subspan(m_body_start);
|
|
m_chunk_pos += chunk_size;
|
|
if (chunk_size == 0)
|
|
{
|
|
TORRENT_ASSERT(m_recv_buffer.get().size() < m_chunk_pos + m_body_start + 1
|
|
|| m_recv_buffer.get()[static_cast<std::ptrdiff_t>(m_chunk_pos + m_body_start)] == 'H'
|
|
|| (m_parser.chunked_encoding()
|
|
&& m_recv_buffer.get()[static_cast<std::ptrdiff_t>(m_chunk_pos + m_body_start)] == '\r'));
|
|
m_chunk_pos = -1;
|
|
}
|
|
}
|
|
}
|
|
|
|
int payload = int(bytes_transferred);
|
|
if (payload > m_response_left) payload = int(m_response_left);
|
|
if (payload > front_request.length) payload = front_request.length;
|
|
received_bytes(payload, 0);
|
|
incoming_piece_fragment(payload);
|
|
m_response_left -= payload;
|
|
|
|
if (m_parser.status_code() == 503)
|
|
{
|
|
if (!m_parser.finished()) return;
|
|
|
|
int retry_time = std::atoi(std::string(recv_buffer.begin(), recv_buffer.end()).c_str());
|
|
if (retry_time <= 0) retry_time = 60;
|
|
#ifndef TORRENT_DISABLE_LOGGING
|
|
peer_log(peer_log_alert::info, "CONNECT", "retrying in %d seconds", retry_time);
|
|
#endif
|
|
|
|
received_bytes(0, int(bytes_transferred));
|
|
// temporarily unavailable, retry later
|
|
t->retry_web_seed(this, seconds32(retry_time));
|
|
disconnect(error_code(m_parser.status_code(), http_category()), operation_t::bittorrent, failure);
|
|
return;
|
|
}
|
|
|
|
|
|
// we only received the header, no data
|
|
if (recv_buffer.empty()) break;
|
|
|
|
if (recv_buffer.size() < front_request.length) break;
|
|
|
|
// if the response is chunked, we need to receive the last
|
|
// terminating chunk and the tail headers before we can proceed
|
|
if (m_parser.chunked_encoding() && m_chunk_pos >= 0) break;
|
|
|
|
m_requests.pop_front();
|
|
incoming_piece(front_request, recv_buffer.begin());
|
|
if (associated_torrent().expired()) return;
|
|
|
|
int const size_to_cut = m_body_start + front_request.length;
|
|
TORRENT_ASSERT(m_recv_buffer.get().size() < size_to_cut + 1
|
|
|| m_recv_buffer.get()[size_to_cut] == 'H'
|
|
|| (m_parser.chunked_encoding() && m_recv_buffer.get()[size_to_cut] == '\r'));
|
|
|
|
m_recv_buffer.cut(size_to_cut, t->block_size() + 1024);
|
|
if (m_response_left == 0) m_chunk_pos = 0;
|
|
else m_chunk_pos -= front_request.length;
|
|
TORRENT_ASSERT(bytes_transferred >= aux::numeric_cast<std::size_t>(payload));
|
|
bytes_transferred -= aux::numeric_cast<std::size_t>(payload);
|
|
m_body_start = 0;
|
|
if (m_response_left > 0) continue;
|
|
TORRENT_ASSERT(m_response_left == 0);
|
|
m_parser.reset();
|
|
}
|
|
}
|
|
|
|
void http_seed_connection::get_specific_peer_info(peer_info& p) const
|
|
{
|
|
web_connection_base::get_specific_peer_info(p);
|
|
p.flags |= peer_info::local_connection;
|
|
p.connection_type = peer_info::http_seed;
|
|
}
|
|
|
|
}
|