fix example of custom storage
This commit is contained in:
parent
e08dc0b21d
commit
f67fb0850f
|
@ -2,6 +2,7 @@ project(libtorrent-examples)
|
||||||
|
|
||||||
set(single_file_examples
|
set(single_file_examples
|
||||||
simple_client
|
simple_client
|
||||||
|
custom_storage
|
||||||
stats_counters
|
stats_counters
|
||||||
dump_torrent
|
dump_torrent
|
||||||
make_torrent
|
make_torrent
|
||||||
|
|
|
@ -18,6 +18,7 @@ project client_test
|
||||||
# C4268: 'identifier' : 'const' static/global data initialized
|
# C4268: 'identifier' : 'const' static/global data initialized
|
||||||
# with compiler generated default constructor fills the object with zeros
|
# with compiler generated default constructor fills the object with zeros
|
||||||
<toolset>msvc:<cflags>/wd4268
|
<toolset>msvc:<cflags>/wd4268
|
||||||
|
<toolset>msvc:<cflags>/wd4373
|
||||||
: default-build
|
: default-build
|
||||||
<link>static
|
<link>static
|
||||||
;
|
;
|
||||||
|
@ -25,6 +26,7 @@ project client_test
|
||||||
exe client_test : client_test.cpp print.cpp torrent_view.cpp session_view.cpp ;
|
exe client_test : client_test.cpp print.cpp torrent_view.cpp session_view.cpp ;
|
||||||
|
|
||||||
exe simple_client : simple_client.cpp ;
|
exe simple_client : simple_client.cpp ;
|
||||||
|
exe custom_storage : custom_storage.cpp ;
|
||||||
exe bt-get : bt-get.cpp ;
|
exe bt-get : bt-get.cpp ;
|
||||||
exe bt-get2 : bt-get2.cpp ;
|
exe bt-get2 : bt-get2.cpp ;
|
||||||
exe stats_counters : stats_counters.cpp ;
|
exe stats_counters : stats_counters.cpp ;
|
||||||
|
|
|
@ -4,6 +4,7 @@ example_programs = \
|
||||||
dump_torrent \
|
dump_torrent \
|
||||||
make_torrent \
|
make_torrent \
|
||||||
simple_client \
|
simple_client \
|
||||||
|
custom_storage \
|
||||||
upnp_test \
|
upnp_test \
|
||||||
bt_get \
|
bt_get \
|
||||||
bt_get2 \
|
bt_get2 \
|
||||||
|
@ -23,6 +24,7 @@ bt_get2_SOURCES = bt-get2.cpp
|
||||||
dump_torrent_SOURCES = dump_torrent.cpp
|
dump_torrent_SOURCES = dump_torrent.cpp
|
||||||
make_torrent_SOURCES = make_torrent.cpp
|
make_torrent_SOURCES = make_torrent.cpp
|
||||||
simple_client_SOURCES = simple_client.cpp
|
simple_client_SOURCES = simple_client.cpp
|
||||||
|
custom_storage_SOURCES = custom_storage.cpp
|
||||||
connection_tester_SOURCES = connection_tester.cpp
|
connection_tester_SOURCES = connection_tester.cpp
|
||||||
upnp_test_SOURCES = upnp_test.cpp
|
upnp_test_SOURCES = upnp_test.cpp
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,125 @@
|
||||||
|
/*
|
||||||
|
|
||||||
|
Copyright (c) 2019, 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 <cstdlib>
|
||||||
|
#include "libtorrent/entry.hpp"
|
||||||
|
#include "libtorrent/bencode.hpp"
|
||||||
|
#include "libtorrent/session.hpp"
|
||||||
|
#include "libtorrent/torrent_info.hpp"
|
||||||
|
#include "libtorrent/fwd.hpp"
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
// -- example begin
|
||||||
|
struct temp_storage : lt::storage_interface
|
||||||
|
{
|
||||||
|
explicit temp_storage(lt::file_storage const& fs) : lt::storage_interface(fs) {}
|
||||||
|
void initialize(lt::storage_error&) override {}
|
||||||
|
bool has_any_file(lt::storage_error&) override { return false; }
|
||||||
|
virtual void set_file_priority(lt::aux::vector<lt::download_priority_t, lt::file_index_t>&
|
||||||
|
, lt::storage_error&) override {}
|
||||||
|
int readv(lt::span<lt::iovec_t const> bufs, lt::piece_index_t piece
|
||||||
|
, int offset, lt::open_mode_t, lt::storage_error&) override
|
||||||
|
{
|
||||||
|
auto const i = m_file_data.find(piece);
|
||||||
|
if (i == m_file_data.end()) return 0;
|
||||||
|
if (int(i->second.size()) <= offset) return 0;
|
||||||
|
lt::iovec_t data{ i->second.data() + offset, int(i->second.size() - offset) };
|
||||||
|
int ret = 0;
|
||||||
|
for (lt::iovec_t const& b : bufs) {
|
||||||
|
int const to_copy = std::min(int(b.size()), int(data.size()));
|
||||||
|
memcpy(b.data(), data.data(), to_copy);
|
||||||
|
data = data.subspan(to_copy);
|
||||||
|
ret += to_copy;
|
||||||
|
if (data.empty()) break;
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
int writev(lt::span<lt::iovec_t const> bufs
|
||||||
|
, lt::piece_index_t const piece, int offset, lt::open_mode_t, lt::storage_error&) override
|
||||||
|
{
|
||||||
|
auto& data = m_file_data[piece];
|
||||||
|
int ret = 0;
|
||||||
|
for (auto& b : bufs) {
|
||||||
|
if (int(data.size()) < offset + b.size()) data.resize(offset + b.size());
|
||||||
|
std::memcpy(data.data() + offset, b.data(), b.size());
|
||||||
|
offset += int(b.size());
|
||||||
|
ret += int(b.size());
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
void rename_file(lt::file_index_t, std::string const&, lt::storage_error&) override
|
||||||
|
{ assert(false); }
|
||||||
|
lt::status_t move_storage(std::string const&
|
||||||
|
, lt::move_flags_t, lt::storage_error&) override { return lt::status_t::no_error; }
|
||||||
|
bool verify_resume_data(lt::add_torrent_params const&
|
||||||
|
, lt::aux::vector<std::string, lt::file_index_t> const&
|
||||||
|
, lt::storage_error&) override
|
||||||
|
{ return false; }
|
||||||
|
void release_files(lt::storage_error&) override {}
|
||||||
|
void delete_files(lt::remove_flags_t, lt::storage_error&) override {}
|
||||||
|
|
||||||
|
std::map<lt::piece_index_t, std::vector<char>> m_file_data;
|
||||||
|
};
|
||||||
|
|
||||||
|
lt::storage_interface* temp_storage_constructor(lt::storage_params const& params, lt::file_pool&)
|
||||||
|
{
|
||||||
|
return new temp_storage(params.files);
|
||||||
|
}
|
||||||
|
|
||||||
|
// -- example end
|
||||||
|
|
||||||
|
int main(int argc, char* argv[]) try
|
||||||
|
{
|
||||||
|
if (argc != 2) {
|
||||||
|
std::cerr << "usage: ./custom_storage torrent-file\n"
|
||||||
|
"to stop the client, press return.\n";
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
lt::session s;
|
||||||
|
lt::add_torrent_params p;
|
||||||
|
p.storage = temp_storage_constructor;
|
||||||
|
p.save_path = "./";
|
||||||
|
p.ti = std::make_shared<lt::torrent_info>(argv[1]);
|
||||||
|
s.add_torrent(p);
|
||||||
|
|
||||||
|
// wait for the user to end
|
||||||
|
char a;
|
||||||
|
int ret = std::scanf("%c\n", &a);
|
||||||
|
(void)ret; // ignore
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
catch (std::exception const& e) {
|
||||||
|
std::cerr << "ERROR: " << e.what() << "\n";
|
||||||
|
}
|
||||||
|
|
|
@ -67,64 +67,11 @@ POSSIBILITY OF SUCH DAMAGE.
|
||||||
// ``std::map``, i.e. in RAM. It's not necessarily very useful in practice, but
|
// ``std::map``, i.e. in RAM. It's not necessarily very useful in practice, but
|
||||||
// illustrates the basics of implementing a custom storage.
|
// illustrates the basics of implementing a custom storage.
|
||||||
//
|
//
|
||||||
// .. code:: c++
|
// .. include:: ../examples/custom_storage.cpp
|
||||||
//
|
// :code: c++
|
||||||
// struct temp_storage : storage_interface
|
// :tab-width: 2
|
||||||
// {
|
// :start-after: -- example begin
|
||||||
// temp_storage(file_storage const& fs) : storage_interface(fs) {}
|
// :end-before: // -- example end
|
||||||
// bool initialize(storage_error& se) override { return false; }
|
|
||||||
// bool has_any_file() override { return false; }
|
|
||||||
// int read(char* buf, int piece, int offset, int size) override
|
|
||||||
// {
|
|
||||||
// std::map<int, std::vector<char>>::const_iterator i = m_file_data.find(piece);
|
|
||||||
// if (i == m_file_data.end()) return 0;
|
|
||||||
// int available = i->second.size() - offset;
|
|
||||||
// if (available <= 0) return 0;
|
|
||||||
// if (available > size) available = size;
|
|
||||||
// memcpy(buf, &i->second[offset], available);
|
|
||||||
// return available;
|
|
||||||
// }
|
|
||||||
// int write(const char* buf, int piece, int offset, int size) override
|
|
||||||
// {
|
|
||||||
// std::vector<char>& data = m_file_data[piece];
|
|
||||||
// if (data.size() < offset + size) data.resize(offset + size);
|
|
||||||
// std::memcpy(&data[offset], buf, size);
|
|
||||||
// return size;
|
|
||||||
// }
|
|
||||||
// bool rename_file(file_index_t file, std::string const& new_name) override
|
|
||||||
// { assert(false); return false; }
|
|
||||||
// status_t move_storage(std::string const& save_path) override { return false; }
|
|
||||||
// bool verify_resume_data(add_torrent_params const& rd
|
|
||||||
// , std::vector<std::string> const* links
|
|
||||||
// , storage_error& error) override { return false; }
|
|
||||||
// std::int64_t physical_offset(int piece, int offset) override
|
|
||||||
// { return piece * files().piece_length() + offset; };
|
|
||||||
// sha1_hash hash_for_slot(int piece, partial_hash& ph, int piece_size) override
|
|
||||||
// {
|
|
||||||
// int left = piece_size - ph.offset;
|
|
||||||
// assert(left >= 0);
|
|
||||||
// if (left > 0)
|
|
||||||
// {
|
|
||||||
// std::vector<char>& data = m_file_data[piece];
|
|
||||||
// // if there are padding files, those blocks will be considered
|
|
||||||
// // completed even though they haven't been written to the storage.
|
|
||||||
// // in this case, just extend the piece buffer to its full size
|
|
||||||
// // and fill it with zeros.
|
|
||||||
// if (data.size() < piece_size) data.resize(piece_size, 0);
|
|
||||||
// ph.h.update(&data[ph.offset], left);
|
|
||||||
// }
|
|
||||||
// return ph.h.final();
|
|
||||||
// }
|
|
||||||
// bool release_files() override { return false; }
|
|
||||||
// bool delete_files() override { return false; }
|
|
||||||
//
|
|
||||||
// std::map<int, std::vector<char>> m_file_data;
|
|
||||||
// };
|
|
||||||
//
|
|
||||||
// storage_interface* temp_storage_constructor(storage_params const& params)
|
|
||||||
// {
|
|
||||||
// return new temp_storage(*params.files);
|
|
||||||
// }
|
|
||||||
namespace libtorrent {
|
namespace libtorrent {
|
||||||
|
|
||||||
namespace aux { struct session_settings; }
|
namespace aux { struct session_settings; }
|
||||||
|
|
Loading…
Reference in New Issue