2007-07-04 08:24:30 +00:00
|
|
|
/*
|
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
Copyright (c) 2008, Arvid Norberg
|
2007-07-04 08:24:30 +00:00
|
|
|
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/pch.hpp"
|
|
|
|
|
|
|
|
#include <ctime>
|
|
|
|
#include <iostream>
|
|
|
|
#include <fstream>
|
|
|
|
#include <iomanip>
|
|
|
|
#include <iterator>
|
|
|
|
#include <algorithm>
|
|
|
|
#include <set>
|
|
|
|
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
#pragma warning(push, 1)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <boost/lexical_cast.hpp>
|
|
|
|
#include <boost/filesystem/path.hpp>
|
2008-05-20 06:23:48 +00:00
|
|
|
#include <boost/filesystem.hpp>
|
2007-07-04 08:24:30 +00:00
|
|
|
#include <boost/next_prior.hpp>
|
|
|
|
#include <boost/bind.hpp>
|
|
|
|
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
#pragma warning(pop)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "libtorrent/torrent_info.hpp"
|
|
|
|
#include "libtorrent/bencode.hpp"
|
|
|
|
#include "libtorrent/hasher.hpp"
|
|
|
|
#include "libtorrent/entry.hpp"
|
|
|
|
|
|
|
|
namespace gr = boost::gregorian;
|
|
|
|
|
|
|
|
using namespace libtorrent;
|
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
|
|
|
|
namespace fs = boost::filesystem;
|
|
|
|
|
|
|
|
void convert_to_utf8(std::string& str, unsigned char chr)
|
|
|
|
{
|
|
|
|
str += 0xc0 | ((chr & 0xff) >> 6);
|
|
|
|
str += 0x80 | (chr & 0x3f);
|
|
|
|
}
|
|
|
|
|
|
|
|
void verify_encoding(file_entry& target)
|
|
|
|
{
|
|
|
|
std::string tmp_path;
|
|
|
|
std::string file_path = target.path.string();
|
|
|
|
bool valid_encoding = true;
|
|
|
|
for (std::string::iterator i = file_path.begin()
|
|
|
|
, end(file_path.end()); i != end; ++i)
|
|
|
|
{
|
|
|
|
// valid ascii-character
|
|
|
|
if ((*i & 0x80) == 0)
|
|
|
|
{
|
|
|
|
tmp_path += *i;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (std::distance(i, end) < 2)
|
|
|
|
{
|
|
|
|
convert_to_utf8(tmp_path, *i);
|
|
|
|
valid_encoding = false;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// valid 2-byte utf-8 character
|
|
|
|
if ((i[0] & 0xe0) == 0xc0
|
|
|
|
&& (i[1] & 0xc0) == 0x80)
|
|
|
|
{
|
|
|
|
tmp_path += i[0];
|
|
|
|
tmp_path += i[1];
|
|
|
|
i += 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (std::distance(i, end) < 3)
|
|
|
|
{
|
|
|
|
convert_to_utf8(tmp_path, *i);
|
|
|
|
valid_encoding = false;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// valid 3-byte utf-8 character
|
|
|
|
if ((i[0] & 0xf0) == 0xe0
|
|
|
|
&& (i[1] & 0xc0) == 0x80
|
|
|
|
&& (i[2] & 0xc0) == 0x80)
|
|
|
|
{
|
|
|
|
tmp_path += i[0];
|
|
|
|
tmp_path += i[1];
|
|
|
|
tmp_path += i[2];
|
|
|
|
i += 2;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (std::distance(i, end) < 4)
|
|
|
|
{
|
|
|
|
convert_to_utf8(tmp_path, *i);
|
|
|
|
valid_encoding = false;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// valid 4-byte utf-8 character
|
|
|
|
if ((i[0] & 0xf0) == 0xe0
|
|
|
|
&& (i[1] & 0xc0) == 0x80
|
|
|
|
&& (i[2] & 0xc0) == 0x80
|
|
|
|
&& (i[3] & 0xc0) == 0x80)
|
|
|
|
{
|
|
|
|
tmp_path += i[0];
|
|
|
|
tmp_path += i[1];
|
|
|
|
tmp_path += i[2];
|
|
|
|
tmp_path += i[3];
|
|
|
|
i += 3;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
convert_to_utf8(tmp_path, *i);
|
|
|
|
valid_encoding = false;
|
|
|
|
}
|
|
|
|
// the encoding was not valid utf-8
|
|
|
|
// save the original encoding and replace the
|
|
|
|
// commonly used path with the correctly
|
|
|
|
// encoded string
|
|
|
|
if (!valid_encoding)
|
|
|
|
{
|
|
|
|
target.orig_path.reset(new fs::path(target.path));
|
|
|
|
target.path = tmp_path;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
bool extract_single_file(lazy_entry const& dict, file_entry& target
|
2007-07-04 08:24:30 +00:00
|
|
|
, std::string const& root_dir)
|
|
|
|
{
|
2008-05-20 06:23:48 +00:00
|
|
|
lazy_entry const* length = dict.dict_find("length");
|
|
|
|
if (length == 0 || length->type() != lazy_entry::int_t)
|
2008-04-16 01:15:06 +00:00
|
|
|
return false;
|
2008-05-20 06:23:48 +00:00
|
|
|
target.size = length->int_value();
|
2007-07-04 08:24:30 +00:00
|
|
|
target.path = root_dir;
|
2007-11-28 03:29:53 +00:00
|
|
|
target.file_base = 0;
|
2007-07-04 08:24:30 +00:00
|
|
|
|
|
|
|
// prefer the name.utf-8
|
|
|
|
// because if it exists, it is more
|
|
|
|
// likely to be correctly encoded
|
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
lazy_entry const* p = dict.dict_find("path.utf-8");
|
|
|
|
if (p == 0 || p->type() != lazy_entry::list_t)
|
|
|
|
p = dict.dict_find("path");
|
|
|
|
if (p == 0 || p->type() != lazy_entry::list_t)
|
|
|
|
return false;
|
2007-07-04 08:24:30 +00:00
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
for (int i = 0, end(p->list_size()); i < end; ++i)
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
2008-05-20 06:23:48 +00:00
|
|
|
if (p->list_at(i)->type() != lazy_entry::string_t)
|
2008-04-16 01:15:06 +00:00
|
|
|
return false;
|
2008-05-20 06:23:48 +00:00
|
|
|
std::string path_element = p->list_at(i)->string_value();
|
|
|
|
if (path_element != "..")
|
|
|
|
target.path /= path_element;
|
2007-07-04 08:24:30 +00:00
|
|
|
}
|
|
|
|
verify_encoding(target);
|
2008-04-16 01:15:06 +00:00
|
|
|
if (target.path.is_complete())
|
|
|
|
return false;
|
|
|
|
return true;
|
2007-07-04 08:24:30 +00:00
|
|
|
}
|
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
bool extract_files(lazy_entry const& list, std::vector<file_entry>& target
|
2007-07-04 08:24:30 +00:00
|
|
|
, std::string const& root_dir)
|
|
|
|
{
|
|
|
|
size_type offset = 0;
|
2008-05-20 06:23:48 +00:00
|
|
|
if (list.type() != lazy_entry::list_t) return false;
|
|
|
|
for (int i = 0, end(list.list_size()); i < end; ++i)
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
|
|
|
target.push_back(file_entry());
|
2008-05-20 06:23:48 +00:00
|
|
|
if (!extract_single_file(*list.list_at(i), target.back(), root_dir))
|
2008-04-16 01:15:06 +00:00
|
|
|
return false;
|
2007-07-04 08:24:30 +00:00
|
|
|
target.back().offset = offset;
|
|
|
|
offset += target.back().size;
|
|
|
|
}
|
2008-04-16 01:15:06 +00:00
|
|
|
return true;
|
2007-07-04 08:24:30 +00:00
|
|
|
}
|
|
|
|
/*
|
|
|
|
void remove_dir(fs::path& p)
|
|
|
|
{
|
2007-10-06 04:22:06 +00:00
|
|
|
TORRENT_ASSERT(p.begin() != p.end());
|
2007-07-04 08:24:30 +00:00
|
|
|
path tmp;
|
|
|
|
for (path::iterator i = boost::next(p.begin()); i != p.end(); ++i)
|
|
|
|
tmp /= *i;
|
|
|
|
p = tmp;
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace libtorrent
|
|
|
|
{
|
|
|
|
|
|
|
|
// standard constructor that parses a torrent file
|
2008-05-20 06:23:48 +00:00
|
|
|
torrent_info::torrent_info(entry const& torrent_file)
|
2007-07-04 08:24:30 +00:00
|
|
|
: m_num_pieces(0)
|
|
|
|
, m_creation_date(pt::ptime(pt::not_a_date_time))
|
|
|
|
, m_multifile(false)
|
|
|
|
, m_private(false)
|
2008-05-20 06:23:48 +00:00
|
|
|
, m_info_section_size(0)
|
|
|
|
, m_piece_hashes(0)
|
|
|
|
{
|
|
|
|
std::vector<char> tmp;
|
|
|
|
std::back_insert_iterator<std::vector<char> > out(tmp);
|
|
|
|
bencode(out, torrent_file);
|
|
|
|
|
|
|
|
lazy_entry e;
|
|
|
|
lazy_bdecode(&tmp[0], &tmp[0] + tmp.size(), e);
|
|
|
|
std::string error;
|
|
|
|
#ifndef BOOST_NO_EXCEPTIONS
|
|
|
|
if (!parse_torrent_file(e, error))
|
|
|
|
throw invalid_torrent_file();
|
|
|
|
#else
|
|
|
|
read_torrent_info(e, error);
|
2007-07-04 08:24:30 +00:00
|
|
|
#endif
|
2008-05-20 06:23:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
torrent_info::torrent_info(lazy_entry const& torrent_file)
|
|
|
|
: m_num_pieces(0)
|
|
|
|
, m_creation_date(pt::ptime(pt::not_a_date_time))
|
|
|
|
, m_multifile(false)
|
|
|
|
, m_private(false)
|
|
|
|
, m_info_section_size(0)
|
|
|
|
, m_piece_hashes(0)
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
2008-04-16 01:15:06 +00:00
|
|
|
std::string error;
|
|
|
|
#ifndef BOOST_NO_EXCEPTIONS
|
2008-05-20 06:23:48 +00:00
|
|
|
if (!parse_torrent_file(torrent_file, error))
|
2007-07-04 08:24:30 +00:00
|
|
|
throw invalid_torrent_file();
|
2008-04-16 01:15:06 +00:00
|
|
|
#else
|
|
|
|
read_torrent_info(torrent_file, error);
|
|
|
|
#endif
|
2007-07-04 08:24:30 +00:00
|
|
|
}
|
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
torrent_info::torrent_info(char const* buffer, int size)
|
|
|
|
: m_num_pieces(0)
|
|
|
|
, m_creation_date(pt::ptime(pt::not_a_date_time))
|
|
|
|
, m_multifile(false)
|
|
|
|
, m_private(false)
|
|
|
|
, m_info_section_size(0)
|
|
|
|
, m_piece_hashes(0)
|
|
|
|
{
|
|
|
|
std::string error;
|
|
|
|
lazy_entry e;
|
|
|
|
lazy_bdecode(buffer, buffer + size, e);
|
|
|
|
#ifndef BOOST_NO_EXCEPTIONS
|
|
|
|
if (!parse_torrent_file(e, error))
|
|
|
|
throw invalid_torrent_file();
|
|
|
|
#else
|
|
|
|
read_torrent_info(e, error);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2007-07-04 08:24:30 +00:00
|
|
|
// constructor used for creating new torrents
|
|
|
|
// will not contain any hashes, comments, creation date
|
|
|
|
// just the necessary to use it with piece manager
|
|
|
|
// used for torrents with no metadata
|
|
|
|
torrent_info::torrent_info(sha1_hash const& info_hash)
|
|
|
|
: m_piece_length(0)
|
|
|
|
, m_total_size(0)
|
|
|
|
, m_num_pieces(0)
|
|
|
|
, m_info_hash(info_hash)
|
|
|
|
, m_name()
|
|
|
|
, m_creation_date(pt::second_clock::universal_time())
|
|
|
|
, m_multifile(false)
|
|
|
|
, m_private(false)
|
2008-05-20 06:23:48 +00:00
|
|
|
, m_info_section_size(0)
|
|
|
|
, m_piece_hashes(0)
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
torrent_info::torrent_info()
|
|
|
|
: m_piece_length(0)
|
|
|
|
, m_total_size(0)
|
|
|
|
, m_num_pieces(0)
|
|
|
|
, m_info_hash(0)
|
|
|
|
, m_name()
|
|
|
|
, m_creation_date(pt::second_clock::universal_time())
|
|
|
|
, m_multifile(false)
|
|
|
|
, m_private(false)
|
2008-05-20 06:23:48 +00:00
|
|
|
, m_info_section_size(0)
|
|
|
|
, m_piece_hashes(0)
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
torrent_info::torrent_info(char const* filename)
|
|
|
|
: m_num_pieces(0)
|
|
|
|
, m_creation_date(pt::ptime(pt::not_a_date_time))
|
|
|
|
, m_multifile(false)
|
|
|
|
, m_private(false)
|
|
|
|
{
|
|
|
|
size_type s = fs::file_size(fs::path(filename));
|
|
|
|
// don't load torrent files larger than 2 MB
|
|
|
|
if (s > 2000000) return;
|
|
|
|
std::vector<char> buf(s);
|
|
|
|
std::ifstream f(filename);
|
|
|
|
f.read(&buf[0], s);
|
|
|
|
|
|
|
|
std::string error;
|
|
|
|
lazy_entry e;
|
|
|
|
lazy_bdecode(&buf[0], &buf[0] + buf.size(), e);
|
|
|
|
#ifndef BOOST_NO_EXCEPTIONS
|
|
|
|
if (!parse_torrent_file(e, error))
|
|
|
|
throw invalid_torrent_file();
|
|
|
|
#else
|
|
|
|
read_torrent_info(e, error);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2007-07-04 08:24:30 +00:00
|
|
|
torrent_info::~torrent_info()
|
|
|
|
{}
|
|
|
|
|
|
|
|
void torrent_info::swap(torrent_info& ti)
|
|
|
|
{
|
|
|
|
using std::swap;
|
|
|
|
m_urls.swap(ti.m_urls);
|
|
|
|
m_url_seeds.swap(ti.m_url_seeds);
|
|
|
|
m_files.swap(ti.m_files);
|
2008-05-20 06:23:48 +00:00
|
|
|
m_files.swap(ti.m_remapped_files);
|
2007-07-04 08:24:30 +00:00
|
|
|
m_nodes.swap(ti.m_nodes);
|
|
|
|
swap(m_num_pieces, ti.m_num_pieces);
|
|
|
|
swap(m_info_hash, ti.m_info_hash);
|
|
|
|
m_name.swap(ti.m_name);
|
|
|
|
swap(m_creation_date, ti.m_creation_date);
|
|
|
|
m_comment.swap(ti.m_comment);
|
|
|
|
m_created_by.swap(ti.m_created_by);
|
|
|
|
swap(m_multifile, ti.m_multifile);
|
|
|
|
swap(m_private, ti.m_private);
|
2008-05-20 06:23:48 +00:00
|
|
|
swap(m_info_section, ti.m_info_section);
|
|
|
|
swap(m_info_section_size, ti.m_info_section_size);
|
|
|
|
swap(m_piece_hashes, ti.m_piece_hashes);
|
|
|
|
swap(m_info_dict, ti.m_info_dict);
|
2007-07-04 08:24:30 +00:00
|
|
|
}
|
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
bool torrent_info::parse_info_section(lazy_entry const& info, std::string& error)
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
2008-05-20 06:23:48 +00:00
|
|
|
if (info.type() != lazy_entry::dict_t)
|
2008-04-16 01:15:06 +00:00
|
|
|
{
|
|
|
|
error = "'info' entry is not a dictionary";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
// hash the info-field to calculate info-hash
|
2007-07-04 08:24:30 +00:00
|
|
|
hasher h;
|
2008-05-20 06:23:48 +00:00
|
|
|
std::pair<char const*, int> section = info.data_section();
|
|
|
|
h.update(section.first, section.second);
|
2007-07-04 08:24:30 +00:00
|
|
|
m_info_hash = h.final();
|
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
// copy the info section
|
|
|
|
m_info_section_size = section.second;
|
|
|
|
m_info_section.reset(new char[m_info_section_size]);
|
|
|
|
memcpy(m_info_section.get(), section.first, m_info_section_size);
|
|
|
|
TORRENT_ASSERT(section.first[0] == 'd');
|
|
|
|
TORRENT_ASSERT(section.first[m_info_section_size-1] == 'e');
|
|
|
|
|
2007-07-04 08:24:30 +00:00
|
|
|
// extract piece length
|
2008-05-20 06:23:48 +00:00
|
|
|
m_piece_length = info.dict_find_int_value("piece length", -1);
|
2008-04-16 01:15:06 +00:00
|
|
|
if (m_piece_length <= 0)
|
|
|
|
{
|
2008-05-20 06:23:48 +00:00
|
|
|
error = "invalid or missing 'piece length' entry in torrent file";
|
2008-04-16 01:15:06 +00:00
|
|
|
return false;
|
|
|
|
}
|
2007-07-04 08:24:30 +00:00
|
|
|
|
|
|
|
// extract file name (or the directory name if it's a multifile libtorrent)
|
2008-05-20 06:23:48 +00:00
|
|
|
m_name = info.dict_find_string_value("name.utf-8");
|
|
|
|
if (m_name.empty()) m_name = info.dict_find_string_value("name");
|
|
|
|
|
|
|
|
if (m_name.empty())
|
2008-04-16 01:15:06 +00:00
|
|
|
{
|
2008-05-20 06:23:48 +00:00
|
|
|
error = "invalid name in torrent file";
|
|
|
|
return false;
|
2008-04-16 01:15:06 +00:00
|
|
|
}
|
2008-05-20 06:23:48 +00:00
|
|
|
|
2007-07-04 08:24:30 +00:00
|
|
|
fs::path tmp = m_name;
|
2008-04-16 01:15:06 +00:00
|
|
|
if (tmp.is_complete())
|
|
|
|
{
|
|
|
|
m_name = tmp.leaf();
|
|
|
|
}
|
|
|
|
else if (tmp.has_branch_path())
|
|
|
|
{
|
|
|
|
fs::path p;
|
|
|
|
for (fs::path::iterator i = tmp.begin()
|
|
|
|
, end(tmp.end()); i != end; ++i)
|
|
|
|
{
|
|
|
|
if (*i == "." || *i == "..") continue;
|
|
|
|
p /= *i;
|
|
|
|
}
|
|
|
|
m_name = p.string();
|
|
|
|
}
|
|
|
|
if (m_name == ".." || m_name == ".")
|
|
|
|
{
|
|
|
|
error = "invalid 'name' of torrent (possible exploit attempt)";
|
|
|
|
return false;
|
|
|
|
}
|
2007-07-04 08:24:30 +00:00
|
|
|
|
|
|
|
// extract file list
|
2008-05-20 06:23:48 +00:00
|
|
|
lazy_entry const* i = info.dict_find_list("files");
|
2007-07-04 08:24:30 +00:00
|
|
|
if (i == 0)
|
|
|
|
{
|
|
|
|
// if there's no list of files, there has to be a length
|
|
|
|
// field.
|
|
|
|
file_entry e;
|
|
|
|
e.path = m_name;
|
|
|
|
e.offset = 0;
|
2008-05-20 06:23:48 +00:00
|
|
|
e.size = info.dict_find_int_value("length", -1);
|
|
|
|
if (e.size < 0)
|
|
|
|
{
|
|
|
|
error = "invalid length of torrent";
|
|
|
|
return false;
|
|
|
|
}
|
2007-07-04 08:24:30 +00:00
|
|
|
m_files.push_back(e);
|
2008-05-20 06:23:48 +00:00
|
|
|
m_multifile = false;
|
2007-07-04 08:24:30 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2008-05-20 06:23:48 +00:00
|
|
|
if (!extract_files(*i, m_files, m_name))
|
2008-04-16 01:15:06 +00:00
|
|
|
{
|
|
|
|
error = "failed to parse files from torrent file";
|
|
|
|
return false;
|
|
|
|
}
|
2007-07-04 08:24:30 +00:00
|
|
|
m_multifile = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// calculate total size of all pieces
|
|
|
|
m_total_size = 0;
|
|
|
|
for (std::vector<file_entry>::iterator i = m_files.begin(); i != m_files.end(); ++i)
|
|
|
|
m_total_size += i->size;
|
|
|
|
|
|
|
|
// extract sha-1 hashes for all pieces
|
|
|
|
// we want this division to round upwards, that's why we have the
|
|
|
|
// extra addition
|
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
m_num_pieces = static_cast<int>((m_total_size + m_piece_length - 1) / m_piece_length);
|
|
|
|
|
|
|
|
lazy_entry const* pieces = info.dict_find("pieces");
|
|
|
|
if (pieces == 0 || pieces->type() != lazy_entry::string_t)
|
2008-04-16 01:15:06 +00:00
|
|
|
{
|
|
|
|
error = "invalid or missing 'pieces' entry in torrent file";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
if (pieces->string_length() != m_num_pieces * 20)
|
2008-04-16 01:15:06 +00:00
|
|
|
{
|
|
|
|
error = "incorrect number of piece hashes in torrent file";
|
|
|
|
return false;
|
|
|
|
}
|
2007-07-04 08:24:30 +00:00
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
m_piece_hashes = m_info_section.get() + (pieces->string_ptr() - section.first);
|
|
|
|
TORRENT_ASSERT(m_piece_hashes >= m_info_section.get());
|
|
|
|
TORRENT_ASSERT(m_piece_hashes < m_info_section.get() + m_info_section_size);
|
2007-07-04 08:24:30 +00:00
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
m_private = info.dict_find_int_value("private", 0);
|
2008-04-16 01:15:06 +00:00
|
|
|
return true;
|
2007-07-04 08:24:30 +00:00
|
|
|
}
|
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
bool torrent_info::parse_torrent_file(lazy_entry const& torrent_file, std::string& error)
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
2008-05-20 06:23:48 +00:00
|
|
|
if (torrent_file.type() != lazy_entry::dict_t)
|
2008-04-16 01:15:06 +00:00
|
|
|
{
|
|
|
|
error = "torrent file is not a dictionary";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-07-04 08:24:30 +00:00
|
|
|
// extract the url of the tracker
|
2008-05-20 06:23:48 +00:00
|
|
|
lazy_entry const* i = torrent_file.dict_find_list("announce-list");
|
|
|
|
if (i)
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
2008-05-20 06:23:48 +00:00
|
|
|
m_urls.reserve(i->list_size());
|
|
|
|
for (int j = 0, end(i->list_size()); j < end; ++j)
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
2008-05-20 06:23:48 +00:00
|
|
|
lazy_entry const* tier = i->list_at(j);
|
|
|
|
if (tier->type() != lazy_entry::list_t) continue;
|
|
|
|
for (int k = 0, end(tier->list_size()); k < end; ++k)
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
2008-05-20 06:23:48 +00:00
|
|
|
announce_entry e(tier->list_string_value_at(k));
|
|
|
|
if (e.url.empty()) continue;
|
|
|
|
e.tier = j;
|
2007-07-04 08:24:30 +00:00
|
|
|
m_urls.push_back(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// shuffle each tier
|
|
|
|
std::vector<announce_entry>::iterator start = m_urls.begin();
|
|
|
|
std::vector<announce_entry>::iterator stop;
|
|
|
|
int current_tier = m_urls.front().tier;
|
|
|
|
for (stop = m_urls.begin(); stop != m_urls.end(); ++stop)
|
|
|
|
{
|
|
|
|
if (stop->tier != current_tier)
|
|
|
|
{
|
|
|
|
std::random_shuffle(start, stop);
|
|
|
|
start = stop;
|
|
|
|
current_tier = stop->tier;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
std::random_shuffle(start, stop);
|
|
|
|
}
|
2008-04-16 01:15:06 +00:00
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
|
|
|
|
if (m_urls.empty())
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
2008-05-20 06:23:48 +00:00
|
|
|
announce_entry e(torrent_file.dict_find_string_value("announce"));
|
|
|
|
if (!e.url.empty()) m_urls.push_back(e);
|
2007-07-04 08:24:30 +00:00
|
|
|
}
|
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
lazy_entry const* nodes = torrent_file.dict_find_list("nodes");
|
|
|
|
if (nodes)
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
2008-05-20 06:23:48 +00:00
|
|
|
for (int i = 0, end(nodes->list_size()); i < end; ++i)
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
2008-05-20 06:23:48 +00:00
|
|
|
lazy_entry const* n = nodes->list_at(i);
|
|
|
|
if (n->type() != lazy_entry::list_t
|
|
|
|
|| n->list_size() < 2
|
|
|
|
|| n->list_at(0)->type() != lazy_entry::string_t
|
|
|
|
|| n->list_at(1)->type() != lazy_entry::int_t)
|
|
|
|
continue;
|
|
|
|
m_nodes.push_back(std::make_pair(
|
|
|
|
n->list_at(0)->string_value()
|
|
|
|
, int(n->list_at(1)->int_value())));
|
2007-07-04 08:24:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// extract creation date
|
2008-05-20 06:23:48 +00:00
|
|
|
size_type cd = torrent_file.dict_find_int_value("creation date", -1);
|
|
|
|
if (cd >= 0)
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
|
|
|
m_creation_date = pt::ptime(gr::date(1970, gr::Jan, 1))
|
2008-05-20 06:23:48 +00:00
|
|
|
+ pt::seconds(long(cd));
|
2007-07-04 08:24:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// if there are any url-seeds, extract them
|
2008-05-20 06:23:48 +00:00
|
|
|
lazy_entry const* url_seeds = torrent_file.dict_find("url-list");
|
|
|
|
if (url_seeds && url_seeds->type() == lazy_entry::string_t)
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
2008-05-20 06:23:48 +00:00
|
|
|
m_url_seeds.push_back(url_seeds->string_value());
|
2008-04-16 01:15:06 +00:00
|
|
|
}
|
2008-05-20 06:23:48 +00:00
|
|
|
else if (url_seeds && url_seeds->type() == lazy_entry::list_t)
|
2008-04-16 01:15:06 +00:00
|
|
|
{
|
2008-05-20 06:23:48 +00:00
|
|
|
for (int i = 0, end(url_seeds->list_size()); i < end; ++i)
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
2008-05-20 06:23:48 +00:00
|
|
|
lazy_entry const* url = url_seeds->list_at(i);
|
|
|
|
if (url->type() != lazy_entry::string_t) continue;
|
|
|
|
m_url_seeds.push_back(url->string_value());
|
2007-07-04 08:24:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
m_comment = torrent_file.dict_find_string_value("comment.utf-8");
|
|
|
|
if (m_comment.empty()) m_comment = torrent_file.dict_find_string_value("comment");
|
2007-07-04 08:24:30 +00:00
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
m_created_by = torrent_file.dict_find_string_value("created by.utf-8");
|
|
|
|
if (m_created_by.empty()) m_created_by = torrent_file.dict_find_string_value("created by");
|
2007-07-04 08:24:30 +00:00
|
|
|
|
2008-05-20 06:23:48 +00:00
|
|
|
lazy_entry const* info = torrent_file.dict_find_dict("info");
|
|
|
|
if (info == 0)
|
2008-04-16 01:15:06 +00:00
|
|
|
{
|
|
|
|
error = "missing or invalid 'info' section in torrent file";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return parse_info_section(*info, error);
|
2007-07-04 08:24:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
boost::optional<pt::ptime>
|
|
|
|
torrent_info::creation_date() const
|
|
|
|
{
|
|
|
|
if (m_creation_date != pt::ptime(gr::date(pt::not_a_date_time)))
|
|
|
|
{
|
|
|
|
return boost::optional<pt::ptime>(m_creation_date);
|
|
|
|
}
|
|
|
|
return boost::optional<pt::ptime>();
|
|
|
|
}
|
|
|
|
|
|
|
|
void torrent_info::add_tracker(std::string const& url, int tier)
|
|
|
|
{
|
|
|
|
announce_entry e(url);
|
|
|
|
e.tier = tier;
|
|
|
|
m_urls.push_back(e);
|
|
|
|
|
|
|
|
using boost::bind;
|
|
|
|
std::sort(m_urls.begin(), m_urls.end(), boost::bind<bool>(std::less<int>()
|
|
|
|
, bind(&announce_entry::tier, _1), bind(&announce_entry::tier, _2)));
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------- start deprecation -------
|
|
|
|
|
|
|
|
void torrent_info::print(std::ostream& os) const
|
|
|
|
{
|
|
|
|
os << "trackers:\n";
|
|
|
|
for (std::vector<announce_entry>::const_iterator i = trackers().begin();
|
|
|
|
i != trackers().end(); ++i)
|
|
|
|
{
|
|
|
|
os << i->tier << ": " << i->url << "\n";
|
|
|
|
}
|
|
|
|
if (!m_comment.empty())
|
|
|
|
os << "comment: " << m_comment << "\n";
|
|
|
|
// if (m_creation_date != pt::ptime(gr::date(pt::not_a_date_time)))
|
|
|
|
// os << "creation date: " << to_simple_string(m_creation_date) << "\n";
|
|
|
|
os << "private: " << (m_private?"yes":"no") << "\n";
|
|
|
|
os << "number of pieces: " << num_pieces() << "\n";
|
|
|
|
os << "piece length: " << piece_length() << "\n";
|
|
|
|
os << "files:\n";
|
|
|
|
for (file_iterator i = begin_files(); i != end_files(); ++i)
|
|
|
|
os << " " << std::setw(11) << i->size << " " << i->path.string() << "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------- end deprecation -------
|
|
|
|
|
2008-04-06 21:12:32 +00:00
|
|
|
int torrent_info::piece_size(int index) const
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
2007-10-06 04:22:06 +00:00
|
|
|
TORRENT_ASSERT(index >= 0 && index < num_pieces());
|
2007-07-04 08:24:30 +00:00
|
|
|
if (index == num_pieces()-1)
|
|
|
|
{
|
2008-04-16 01:15:06 +00:00
|
|
|
int size = int(total_size()
|
|
|
|
- (num_pieces() - 1) * piece_length());
|
2007-10-06 04:22:06 +00:00
|
|
|
TORRENT_ASSERT(size > 0);
|
|
|
|
TORRENT_ASSERT(size <= piece_length());
|
2008-04-06 21:12:32 +00:00
|
|
|
return int(size);
|
2007-07-04 08:24:30 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
return piece_length();
|
|
|
|
}
|
|
|
|
|
2007-11-28 03:29:53 +00:00
|
|
|
bool torrent_info::remap_files(std::vector<file_entry> const& map)
|
2007-09-21 00:55:58 +00:00
|
|
|
{
|
|
|
|
size_type offset = 0;
|
|
|
|
m_remapped_files.resize(map.size());
|
|
|
|
|
|
|
|
for (int i = 0; i < int(map.size()); ++i)
|
|
|
|
{
|
|
|
|
file_entry& fe = m_remapped_files[i];
|
2007-11-28 03:29:53 +00:00
|
|
|
fe.path = map[i].path;
|
2007-09-21 00:55:58 +00:00
|
|
|
fe.offset = offset;
|
2007-11-28 03:29:53 +00:00
|
|
|
fe.size = map[i].size;
|
|
|
|
fe.file_base = map[i].file_base;
|
|
|
|
fe.orig_path.reset();
|
2007-09-21 00:55:58 +00:00
|
|
|
offset += fe.size;
|
|
|
|
}
|
|
|
|
if (offset != total_size())
|
|
|
|
{
|
|
|
|
m_remapped_files.clear();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-11-28 03:29:53 +00:00
|
|
|
#ifndef NDEBUG
|
|
|
|
std::vector<file_entry> map2(m_remapped_files);
|
|
|
|
std::sort(map2.begin(), map2.end()
|
|
|
|
, bind(&file_entry::file_base, _1) < bind(&file_entry::file_base, _2));
|
|
|
|
std::stable_sort(map2.begin(), map2.end()
|
|
|
|
, bind(&file_entry::path, _1) < bind(&file_entry::path, _2));
|
|
|
|
fs::path last_path;
|
|
|
|
size_type last_end = 0;
|
|
|
|
for (std::vector<file_entry>::iterator i = map2.begin(), end(map2.end());
|
|
|
|
i != end; ++i)
|
|
|
|
{
|
|
|
|
if (last_path == i->path)
|
|
|
|
{
|
|
|
|
assert(last_end <= i->file_base);
|
|
|
|
}
|
|
|
|
last_end = i->file_base + i->size;
|
|
|
|
last_path = i->path;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2007-09-21 00:55:58 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2007-07-04 08:24:30 +00:00
|
|
|
std::vector<file_slice> torrent_info::map_block(int piece, size_type offset
|
2008-04-06 21:12:32 +00:00
|
|
|
, int size_, bool storage) const
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
2007-10-06 04:22:06 +00:00
|
|
|
TORRENT_ASSERT(num_files() > 0);
|
2007-07-04 08:24:30 +00:00
|
|
|
std::vector<file_slice> ret;
|
|
|
|
|
|
|
|
size_type start = piece * (size_type)m_piece_length + offset;
|
2008-04-06 21:12:32 +00:00
|
|
|
size_type size = size_;
|
2007-10-06 04:22:06 +00:00
|
|
|
TORRENT_ASSERT(start + size <= m_total_size);
|
2007-07-04 08:24:30 +00:00
|
|
|
|
|
|
|
// find the file iterator and file offset
|
|
|
|
// TODO: make a vector that can map piece -> file index in O(1)
|
|
|
|
size_type file_offset = start;
|
|
|
|
std::vector<file_entry>::const_iterator file_iter;
|
|
|
|
|
|
|
|
int counter = 0;
|
2007-09-21 00:55:58 +00:00
|
|
|
for (file_iter = begin_files(storage);; ++counter, ++file_iter)
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
2007-10-06 04:22:06 +00:00
|
|
|
TORRENT_ASSERT(file_iter != end_files(storage));
|
2007-07-04 08:24:30 +00:00
|
|
|
if (file_offset < file_iter->size)
|
|
|
|
{
|
|
|
|
file_slice f;
|
|
|
|
f.file_index = counter;
|
2007-11-28 03:29:53 +00:00
|
|
|
f.offset = file_offset + file_iter->file_base;
|
2007-07-04 08:24:30 +00:00
|
|
|
f.size = (std::min)(file_iter->size - file_offset, (size_type)size);
|
|
|
|
size -= f.size;
|
|
|
|
file_offset += f.size;
|
|
|
|
ret.push_back(f);
|
|
|
|
}
|
|
|
|
|
2007-10-06 04:22:06 +00:00
|
|
|
TORRENT_ASSERT(size >= 0);
|
2007-07-04 08:24:30 +00:00
|
|
|
if (size <= 0) break;
|
|
|
|
|
|
|
|
file_offset -= file_iter->size;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
peer_request torrent_info::map_file(int file_index, size_type file_offset
|
2007-09-21 00:55:58 +00:00
|
|
|
, int size, bool storage) const
|
2007-07-04 08:24:30 +00:00
|
|
|
{
|
2007-10-06 04:22:06 +00:00
|
|
|
TORRENT_ASSERT(file_index < num_files(storage));
|
|
|
|
TORRENT_ASSERT(file_index >= 0);
|
2007-09-21 00:55:58 +00:00
|
|
|
size_type offset = file_offset + file_at(file_index, storage).offset;
|
2007-07-04 08:24:30 +00:00
|
|
|
|
|
|
|
peer_request ret;
|
2008-04-06 21:12:32 +00:00
|
|
|
ret.piece = int(offset / piece_length());
|
|
|
|
ret.start = int(offset - ret.piece * piece_length());
|
2007-07-04 08:24:30 +00:00
|
|
|
ret.length = size;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2008-04-16 01:15:06 +00:00
|
|
|
|