merge checking fix from RC_0_16
This commit is contained in:
parent
e45124fc22
commit
38d7e58177
|
@ -8,6 +8,7 @@
|
||||||
* fix uTP edge case where udp socket buffer fills up
|
* fix uTP edge case where udp socket buffer fills up
|
||||||
* fix nagle implementation in uTP
|
* fix nagle implementation in uTP
|
||||||
|
|
||||||
|
* fixed files checking issue
|
||||||
* added missing pop_alerts() to python bindings
|
* added missing pop_alerts() to python bindings
|
||||||
* fixed typos in configure script, inversing some feature-enable/disable flags
|
* fixed typos in configure script, inversing some feature-enable/disable flags
|
||||||
* added missing flag_update_subscribe to python bindings
|
* added missing flag_update_subscribe to python bindings
|
||||||
|
|
|
@ -463,7 +463,8 @@ namespace libtorrent
|
||||||
std::vector<boost::uint8_t>().swap(m_file_priority);
|
std::vector<boost::uint8_t>().swap(m_file_priority);
|
||||||
// close files that were opened in write mode
|
// close files that were opened in write mode
|
||||||
m_pool.release(this);
|
m_pool.release(this);
|
||||||
return false;
|
|
||||||
|
return error() ? true : false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void default_storage::finalize_file(int index)
|
void default_storage::finalize_file(int index)
|
||||||
|
@ -2020,6 +2021,7 @@ ret:
|
||||||
{
|
{
|
||||||
error = m_storage->error();
|
error = m_storage->error();
|
||||||
TORRENT_ASSERT(error);
|
TORRENT_ASSERT(error);
|
||||||
|
m_current_slot = 0;
|
||||||
return fatal_disk_error;
|
return fatal_disk_error;
|
||||||
}
|
}
|
||||||
m_state = state_finished;
|
m_state = state_finished;
|
||||||
|
@ -2253,7 +2255,16 @@ ret:
|
||||||
{
|
{
|
||||||
if (m_state == state_none) return check_no_fastresume(error);
|
if (m_state == state_none) return check_no_fastresume(error);
|
||||||
|
|
||||||
TORRENT_ASSERT(int(m_piece_to_slot.size()) == m_files.num_pieces());
|
if (m_piece_to_slot.empty())
|
||||||
|
{
|
||||||
|
m_piece_to_slot.clear();
|
||||||
|
m_piece_to_slot.resize(m_files.num_pieces(), has_no_slot);
|
||||||
|
}
|
||||||
|
if (m_slot_to_piece.empty())
|
||||||
|
{
|
||||||
|
m_slot_to_piece.clear();
|
||||||
|
m_slot_to_piece.resize(m_files.num_pieces(), unallocated);
|
||||||
|
}
|
||||||
|
|
||||||
current_slot = m_current_slot;
|
current_slot = m_current_slot;
|
||||||
have_piece = -1;
|
have_piece = -1;
|
||||||
|
|
|
@ -1727,6 +1727,8 @@ namespace libtorrent
|
||||||
if (ret == piece_manager::fatal_disk_error)
|
if (ret == piece_manager::fatal_disk_error)
|
||||||
{
|
{
|
||||||
handle_disk_error(j);
|
handle_disk_error(j);
|
||||||
|
auto_managed(false);
|
||||||
|
pause();
|
||||||
set_state(torrent_status::queued_for_checking);
|
set_state(torrent_status::queued_for_checking);
|
||||||
std::vector<char>().swap(m_resume_data);
|
std::vector<char>().swap(m_resume_data);
|
||||||
lazy_entry().swap(m_resume_entry);
|
lazy_entry().swap(m_resume_entry);
|
||||||
|
@ -2050,6 +2052,7 @@ namespace libtorrent
|
||||||
#if defined TORRENT_VERBOSE_LOGGING || defined TORRENT_LOGGING || defined TORRENT_ERROR_LOGGING
|
#if defined TORRENT_VERBOSE_LOGGING || defined TORRENT_LOGGING || defined TORRENT_ERROR_LOGGING
|
||||||
debug_log("fatal disk error: (%d) %s", j.error.value(), j.error.message().c_str());
|
debug_log("fatal disk error: (%d) %s", j.error.value(), j.error.message().c_str());
|
||||||
#endif
|
#endif
|
||||||
|
auto_managed(false);
|
||||||
pause();
|
pause();
|
||||||
set_error(j.error, j.error_file);
|
set_error(j.error, j.error_file);
|
||||||
// recalculate auto-managed torrents sooner
|
// recalculate auto-managed torrents sooner
|
||||||
|
@ -6876,7 +6879,6 @@ namespace libtorrent
|
||||||
if (!m_storage) init();
|
if (!m_storage) init();
|
||||||
if (!checking_files && should_check_files())
|
if (!checking_files && should_check_files())
|
||||||
queue_torrent_check();
|
queue_torrent_check();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void torrent::set_error(error_code const& ec, std::string const& error_file)
|
void torrent::set_error(error_code const& ec, std::string const& error_file)
|
||||||
|
|
|
@ -56,6 +56,7 @@ test-suite libtorrent :
|
||||||
[ run test_upnp.cpp ]
|
[ run test_upnp.cpp ]
|
||||||
|
|
||||||
[ run test_tracker.cpp ]
|
[ run test_tracker.cpp ]
|
||||||
|
[ run test_checking.cpp ]
|
||||||
[ run test_web_seed.cpp ]
|
[ run test_web_seed.cpp ]
|
||||||
[ run test_bdecode_performance.cpp ]
|
[ run test_bdecode_performance.cpp ]
|
||||||
[ run test_pe_crypto.cpp ]
|
[ run test_pe_crypto.cpp ]
|
||||||
|
|
|
@ -5,6 +5,7 @@ test_programs = \
|
||||||
test_bdecode_performance \
|
test_bdecode_performance \
|
||||||
test_bencoding \
|
test_bencoding \
|
||||||
test_buffer \
|
test_buffer \
|
||||||
|
test_checking \
|
||||||
test_fast_extension \
|
test_fast_extension \
|
||||||
test_hasher \
|
test_hasher \
|
||||||
test_http_connection \
|
test_http_connection \
|
||||||
|
|
|
@ -221,6 +221,30 @@ boost::intrusive_ptr<T> clone_ptr(boost::intrusive_ptr<T> const& ptr)
|
||||||
return boost::intrusive_ptr<T>(new T(*ptr));
|
return boost::intrusive_ptr<T>(new T(*ptr));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void create_random_files(std::string const& path, const int file_sizes[], int num_files)
|
||||||
|
{
|
||||||
|
error_code ec;
|
||||||
|
char* random_data = (char*)malloc(300000);
|
||||||
|
for (int i = 0; i != num_files; ++i)
|
||||||
|
{
|
||||||
|
std::generate(random_data, random_data + 300000, &std::rand);
|
||||||
|
char filename[200];
|
||||||
|
snprintf(filename, sizeof(filename), "%s/test%d", path.c_str(), i);
|
||||||
|
int to_write = file_sizes[i];
|
||||||
|
file f(filename, file::write_only, ec);
|
||||||
|
size_type offset = 0;
|
||||||
|
while (to_write > 0)
|
||||||
|
{
|
||||||
|
int s = (std::min)(to_write, 300000);
|
||||||
|
file::iovec_t b = { random_data, s};
|
||||||
|
f.writev(offset, &b, 1, ec);
|
||||||
|
offset += s;
|
||||||
|
to_write -= s;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
free(random_data);
|
||||||
|
}
|
||||||
|
|
||||||
boost::intrusive_ptr<torrent_info> create_torrent(std::ostream* file, int piece_size
|
boost::intrusive_ptr<torrent_info> create_torrent(std::ostream* file, int piece_size
|
||||||
, int num_pieces, bool add_tracker, bool encrypted_torrent)
|
, int num_pieces, bool add_tracker, bool encrypted_torrent)
|
||||||
{
|
{
|
||||||
|
|
|
@ -55,6 +55,8 @@ void test_sleep(int millisec);
|
||||||
extern int g_udp_tracker_requests;
|
extern int g_udp_tracker_requests;
|
||||||
extern int g_http_tracker_requests;
|
extern int g_http_tracker_requests;
|
||||||
|
|
||||||
|
void create_random_files(std::string const& path, const int file_sizes[], int num_files);
|
||||||
|
|
||||||
boost::intrusive_ptr<libtorrent::torrent_info> create_torrent(std::ostream* file = 0
|
boost::intrusive_ptr<libtorrent::torrent_info> create_torrent(std::ostream* file = 0
|
||||||
, int piece_size = 16 * 1024, int num_pieces = 13, bool add_tracker = true, bool encrypted = false);
|
, int piece_size = 16 * 1024, int num_pieces = 13, bool add_tracker = true, bool encrypted = false);
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,166 @@
|
||||||
|
/*
|
||||||
|
|
||||||
|
Copyright (c) 2013, 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 "libtorrent/session.hpp"
|
||||||
|
#include "test.hpp"
|
||||||
|
#include "setup_transfer.hpp"
|
||||||
|
#include "libtorrent/create_torrent.hpp"
|
||||||
|
|
||||||
|
#include <fstream>
|
||||||
|
|
||||||
|
void test_checking(bool read_only_files, bool corrupt_files = false)
|
||||||
|
{
|
||||||
|
using namespace libtorrent;
|
||||||
|
|
||||||
|
fprintf(stderr, "==== TEST CHECKING %s%s=====\n"
|
||||||
|
, read_only_files?"read-only-files ":""
|
||||||
|
, corrupt_files?"corrupt ":"");
|
||||||
|
|
||||||
|
// in case the previous run was terminated
|
||||||
|
error_code ec;
|
||||||
|
remove_all("tmp1_checking", ec);
|
||||||
|
if (ec) fprintf(stderr, "ERROR: removing tmp1_checking: (%d) %s\n"
|
||||||
|
, ec.value(), ec.message().c_str());
|
||||||
|
|
||||||
|
create_directory("tmp1_checking", ec);
|
||||||
|
if (ec) fprintf(stderr, "ERROR: creating directory tmp1_checking: (%d) %s\n"
|
||||||
|
, ec.value(), ec.message().c_str());
|
||||||
|
create_directory("tmp1_checking/test_torrent_dir", ec);
|
||||||
|
if (ec) fprintf(stderr, "ERROR: creating directory test_torrent_dir: (%d) %s\n"
|
||||||
|
, ec.value(), ec.message().c_str());
|
||||||
|
|
||||||
|
file_storage fs;
|
||||||
|
std::srand(10);
|
||||||
|
int piece_size = 0x4000;
|
||||||
|
static const int file_sizes[] =
|
||||||
|
{ 5, 16 - 5, 16000, 17, 10, 8000, 8000, 1,1,1,1,1,100,1,1,1,1,100,1,1,1,1,1,1
|
||||||
|
,1,1,1,1,1,1,13,65000,34,75,2,30,400,500,23000,900,43000,400,4300,6, 4};
|
||||||
|
const int num_files = sizeof(file_sizes)/sizeof(file_sizes[0]);
|
||||||
|
create_random_files("tmp1_checking/test_torrent_dir", file_sizes, num_files);
|
||||||
|
|
||||||
|
add_files(fs, "tmp1_checking/test_torrent_dir");
|
||||||
|
libtorrent::create_torrent t(fs, piece_size, 0x4000, libtorrent::create_torrent::optimize);
|
||||||
|
|
||||||
|
// calculate the hash for all pieces
|
||||||
|
set_piece_hashes(t, "tmp1_checking", ec);
|
||||||
|
if (ec) fprintf(stderr, "ERROR: set_piece_hashes: (%d) %s\n"
|
||||||
|
, ec.value(), ec.message().c_str());
|
||||||
|
|
||||||
|
std::vector<char> buf;
|
||||||
|
bencode(std::back_inserter(buf), t.generate());
|
||||||
|
boost::intrusive_ptr<torrent_info> ti(new torrent_info(&buf[0], buf.size(), ec));
|
||||||
|
|
||||||
|
fprintf(stderr, "generated torrent: %s tmp1_checking/test_torrent_dir\n"
|
||||||
|
, to_hex(ti->info_hash().to_string()).c_str());
|
||||||
|
|
||||||
|
// overwrite the files with new random data
|
||||||
|
if (corrupt_files)
|
||||||
|
{
|
||||||
|
// increase the size of some files. When they're read only that forces
|
||||||
|
// the checker to open them in write-mode to truncate them
|
||||||
|
static const int file_sizes2[] =
|
||||||
|
{ 5, 16 - 5, 16001, 30, 10, 8000, 8000, 1,1,1,1,1,100,1,1,1,1,100,1,1,1,1,1,1
|
||||||
|
,1,1,1,1,1,1,13,65000,34,75,2,30,400,500,23000,900,43000,400,4300,6, 4};
|
||||||
|
create_random_files("tmp1_checking/test_torrent_dir", file_sizes2, num_files);
|
||||||
|
}
|
||||||
|
|
||||||
|
// make the files read only
|
||||||
|
if (read_only_files)
|
||||||
|
{
|
||||||
|
for (int i = 0; i < num_files; ++i)
|
||||||
|
{
|
||||||
|
char path[1024];
|
||||||
|
snprintf(path, sizeof(path), "tmp1_checking/test_torrent_dir/test%d", i);
|
||||||
|
#ifdef TORRENT_WINDOWS
|
||||||
|
SetFileAttributes(path, FILE_ATTRIBUTE_READONLY);
|
||||||
|
#else
|
||||||
|
chmod(path, S_IRUSR);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
session ses1(fingerprint("LT", 0, 1, 0, 0), std::make_pair(48000, 49000), "0.0.0.0", 0);
|
||||||
|
ses1.set_alert_mask(alert::all_categories);
|
||||||
|
|
||||||
|
add_torrent_params p;
|
||||||
|
p.save_path = "tmp1_checking";
|
||||||
|
p.ti = ti;
|
||||||
|
torrent_handle tor1 = ses1.add_torrent(p, ec);
|
||||||
|
TEST_CHECK(!ec);
|
||||||
|
|
||||||
|
torrent_status st;
|
||||||
|
for (int i = 0; i < 5; ++i)
|
||||||
|
{
|
||||||
|
print_alerts(ses1, "ses1");
|
||||||
|
|
||||||
|
st = tor1.status();
|
||||||
|
|
||||||
|
printf("%d %f %s\n", st.state, st.progress_ppm / 10000.f, st.error.c_str());
|
||||||
|
|
||||||
|
if (st.state != torrent_status::queued_for_checking
|
||||||
|
&& st.state != torrent_status::checking_files
|
||||||
|
&& st.state != torrent_status::checking_resume_data)
|
||||||
|
break;
|
||||||
|
|
||||||
|
if (!st.error.empty()) break;
|
||||||
|
test_sleep(1000);
|
||||||
|
}
|
||||||
|
if (corrupt_files)
|
||||||
|
{
|
||||||
|
TEST_CHECK(!st.is_seeding);
|
||||||
|
TEST_CHECK(!st.error.empty());
|
||||||
|
// wait a while to make sure libtorrent survived the error
|
||||||
|
test_sleep(5000);
|
||||||
|
|
||||||
|
st = tor1.status();
|
||||||
|
TEST_CHECK(!st.is_seeding);
|
||||||
|
TEST_CHECK(!st.error.empty());
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
TEST_CHECK(st.is_seeding);
|
||||||
|
}
|
||||||
|
|
||||||
|
remove_all("tmp1_checking", ec);
|
||||||
|
if (ec) fprintf(stderr, "ERROR: removing tmp1_checking: (%d) %s\n"
|
||||||
|
, ec.value(), ec.message().c_str());
|
||||||
|
}
|
||||||
|
|
||||||
|
int test_main()
|
||||||
|
{
|
||||||
|
test_checking(false);
|
||||||
|
test_checking(true);
|
||||||
|
test_checking(true, true);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
|
@ -225,27 +225,8 @@ int run_suite(char const* protocol, bool test_url_seed, bool chunked_encoding, b
|
||||||
|
|
||||||
if (test_url_seed)
|
if (test_url_seed)
|
||||||
{
|
{
|
||||||
char* random_data = (char*)malloc(300000);
|
create_random_files("tmp1_web_seed/test_torrent_dir", file_sizes, sizeof(file_sizes)/sizeof(file_sizes[0]));
|
||||||
for (int i = 0; i != sizeof(file_sizes)/sizeof(file_sizes[0]); ++i)
|
|
||||||
{
|
|
||||||
std::generate(random_data, random_data + 300000, &std::rand);
|
|
||||||
char filename[200];
|
|
||||||
snprintf(filename, sizeof(filename), "tmp1_web_seed/test_torrent_dir/test%d", i);
|
|
||||||
int to_write = file_sizes[i];
|
|
||||||
file f(filename, file::write_only, ec);
|
|
||||||
size_type offset = 0;
|
|
||||||
while (to_write > 0)
|
|
||||||
{
|
|
||||||
int s = (std::min)(to_write, 300000);
|
|
||||||
file::iovec_t b = { random_data, s};
|
|
||||||
f.writev(offset, &b, 1, ec);
|
|
||||||
offset += s;
|
|
||||||
to_write -= s;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
add_files(fs, "tmp1_web_seed/test_torrent_dir");
|
add_files(fs, "tmp1_web_seed/test_torrent_dir");
|
||||||
free(random_data);
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -301,25 +282,7 @@ int run_suite(char const* protocol, bool test_url_seed, bool chunked_encoding, b
|
||||||
// corrupt the files now, so that the web seed will be banned
|
// corrupt the files now, so that the web seed will be banned
|
||||||
if (test_url_seed)
|
if (test_url_seed)
|
||||||
{
|
{
|
||||||
char* random_data = (char*)malloc(300000);
|
create_random_files("tmp1_web_seed/test_torrent_dir", file_sizes, sizeof(file_sizes)/sizeof(file_sizes[0]));
|
||||||
for (int i = 0; i != sizeof(file_sizes)/sizeof(file_sizes[0]); ++i)
|
|
||||||
{
|
|
||||||
std::generate(random_data, random_data + 300000, &std::rand);
|
|
||||||
char filename[200];
|
|
||||||
snprintf(filename, sizeof(filename), "tmp1_web_seed/test_torrent_dir/test%d", i);
|
|
||||||
int to_write = file_sizes[i];
|
|
||||||
file f(filename, file::write_only, ec);
|
|
||||||
size_type offset = 0;
|
|
||||||
while (to_write > 0)
|
|
||||||
{
|
|
||||||
int s = (std::min)(to_write, 300000);
|
|
||||||
file::iovec_t b = { random_data, s};
|
|
||||||
f.writev(offset, &b, 1, ec);
|
|
||||||
offset += s;
|
|
||||||
to_write -= s;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
free(random_data);
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in New Issue