2008-11-19 19:32:36 +01:00
<?xml version="1.0" encoding="utf-8" ?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
< html xmlns = "http://www.w3.org/1999/xhtml" xml:lang = "en" lang = "en" >
< head >
< meta http-equiv = "Content-Type" content = "text/html; charset=utf-8" / >
2009-08-02 22:09:32 +02:00
< meta name = "generator" content = "Docutils 0.5: http://docutils.sourceforge.net/" / >
2008-11-19 19:32:36 +01:00
< title > creating torrents< / title >
< meta name = "author" content = "Arvid Norberg, arvid@rasterbar.com" / >
< link rel = "stylesheet" type = "text/css" href = "../../css/base.css" / >
< link rel = "stylesheet" type = "text/css" href = "../../css/rst.css" / >
< link rel = "stylesheet" href = "style.css" type = "text/css" / >
< style type = "text/css" >
/* Hides from IE-mac \*/
* html pre { height: 1%; }
/* End hide from IE-mac */
< / style >
< / head >
< body >
< div class = "document" id = "creating-torrents" >
< div id = "container" >
< div id = "headerNav" >
< ul >
< li class = "first" > < a href = "/" > Home< / a > < / li >
< li > < a href = "../../products.html" > Products< / a > < / li >
< li > < a href = "../../contact.html" > Contact< / a > < / li >
< / ul >
< / div >
< div id = "header" >
< h1 > < span > Rasterbar Software< / span > < / h1 >
< h2 > < span > Software developement and consulting< / span > < / h2 >
< / div >
< div id = "main" >
< h1 class = "title" > creating torrents< / h1 >
< table class = "docinfo" frame = "void" rules = "none" >
< col class = "docinfo-name" / >
< col class = "docinfo-content" / >
< tbody valign = "top" >
< tr > < th class = "docinfo-name" > Author:< / th >
2009-03-21 05:36:46 +01:00
< td > Arvid Norberg, < a class = "last reference external" href = "mailto:arvid@rasterbar.com" > arvid@ rasterbar.com< / a > < / td > < / tr >
2008-11-19 19:32:36 +01:00
< / tbody >
< / table >
< div class = "contents topic" id = "table-of-contents" >
< p class = "topic-title first" > Table of contents< / p >
< ul class = "simple" >
2009-11-11 08:03:50 +01:00
< li > < a class = "reference internal" href = "#overview" id = "id3" > overview< / a > < / li >
< li > < a class = "reference internal" href = "#high-level-example" id = "id4" > high level example< / a > < / li >
< li > < a class = "reference internal" href = "#add-files" id = "id5" > add_files< / a > < / li >
< li > < a class = "reference internal" href = "#set-piece-hashes" id = "id6" > set_piece_hashes()< / a > < / li >
< li > < a class = "reference internal" href = "#file-storage" id = "id7" > file_storage< / a > < ul >
< li > < a class = "reference internal" href = "#add-file" id = "id8" > add_file()< / a > < / li >
< li > < a class = "reference internal" href = "#id1" id = "id9" > add_file< / a > < / li >
2009-01-13 22:49:03 +01:00
< / ul >
< / li >
2009-11-11 08:03:50 +01:00
< li > < a class = "reference internal" href = "#create-torrent" id = "id10" > create_torrent< / a > < ul >
< li > < a class = "reference internal" href = "#id2" id = "id11" > create_torrent()< / a > < / li >
< li > < a class = "reference internal" href = "#generate" id = "id12" > generate()< / a > < / li >
< li > < a class = "reference internal" href = "#set-comment" id = "id13" > set_comment()< / a > < / li >
< li > < a class = "reference internal" href = "#set-creator" id = "id14" > set_creator()< / a > < / li >
< li > < a class = "reference internal" href = "#set-hash" id = "id15" > set_hash()< / a > < / li >
< li > < a class = "reference internal" href = "#add-url-seed" id = "id16" > add_url_seed()< / a > < / li >
< li > < a class = "reference internal" href = "#add-node" id = "id17" > add_node()< / a > < / li >
< li > < a class = "reference internal" href = "#add-tracker" id = "id18" > add_tracker()< / a > < / li >
< li > < a class = "reference internal" href = "#set-priv-priv" id = "id19" > set_priv() priv()< / a > < / li >
2008-11-19 19:32:36 +01:00
< / ul >
< / li >
< / ul >
< / div >
< div class = "section" id = "overview" >
< h1 > overview< / h1 >
< p > This section describes the functions and classes that are used
to create torrent files. It is a layered API with low level classes
and higher level convenience functions. A torrent is created in 4
steps:< / p >
< ol class = "arabic simple" >
< li > first the files that will be part of the torrent are determined.< / li >
< li > the torrent properties are set, such as tracker url, web seeds,
DHT nodes etc.< / li >
< li > Read through all the files in the torrent, SHA-1 all the data
and set the piece hashes.< / li >
< li > The torrent is bencoded into a file or buffer.< / li >
< / ol >
< p > If there are a lot of files and or deep directoy hierarchies to
traverse, step one can be time consuming.< / p >
< p > Typically step 3 is by far the most time consuming step, since it
requires to read all the bytes from all the files in the torrent.< / p >
< p > All of these classes and functions are declared by including
< tt class = "docutils literal" > < span class = "pre" > libtorrent/create_torrent.hpp< / span > < / tt > .< / p >
< / div >
< div class = "section" id = "high-level-example" >
< h1 > high level example< / h1 >
< pre class = "literal-block" >
file_storage fs;
// recursively adds files in directories
add_files(fs, " ./my_torrent" );
create_torrent t(fs);
t.add_tracker(" http://my.tracker.com/announce" );
t.set_creator(" libtorrent example" );
// reads the files and calculates the hashes
set_piece_hashes(t, " ." );
ofstream out(" my_torrent.torrent" , std::ios_base::binary);
bencode(std::ostream_iterator< char> (out), t.generate());
< / pre >
< / div >
< div class = "section" id = "add-files" >
< h1 > add_files< / h1 >
< blockquote >
< pre class = "literal-block" >
template < class Pred>
void add_files(file_storage& fs, boost::filesystem::path const& path, Pred p);
2008-12-02 09:41:02 +01:00
template < class Pred>
void add_files(file_storage& fs, boost::filesystem::wpath const& path, Pred p);
2008-11-19 19:32:36 +01:00
void add_files(file_storage& fs, boost::filesystem::path const& path);
2008-12-02 09:41:02 +01:00
void add_files(file_storage& fs, boost::filesystem::wpath const& path);
2008-11-19 19:32:36 +01:00
< / pre >
< / blockquote >
< p > Adds the file specified by < tt class = "docutils literal" > < span class = "pre" > path< / span > < / tt > to the < tt class = "docutils literal" > < span class = "pre" > file_storage< / span > < / tt > object. In case < tt class = "docutils literal" > < span class = "pre" > path< / span > < / tt >
refers to a diretory, files will be added recursively from the directory.< / p >
< p > If specified, the predicate < tt class = "docutils literal" > < span class = "pre" > p< / span > < / tt > is called once for every file and directory that
is encountered. files for which < tt class = "docutils literal" > < span class = "pre" > p< / span > < / tt > returns true are added, and directories for
which < tt class = "docutils literal" > < span class = "pre" > p< / span > < / tt > returns true are traversed. < tt class = "docutils literal" > < span class = "pre" > p< / span > < / tt > must have the following signature:< / p >
< pre class = "literal-block" >
bool Pred(boost::filesystem::path const& p);
< / pre >
2008-12-02 09:41:02 +01:00
< p > and for the wpath version:< / p >
< pre class = "literal-block" >
bool Pred(boost::filesystem::wpath const& p);
< / pre >
2008-11-19 19:32:36 +01:00
< p > The path that is passed in to the predicate is the full path of the file or
directory. If no predicate is specified, all files are added, and all directories
are traveresed.< / p >
< p > The " .." directory is never traversed.< / p >
< / div >
< div class = "section" id = "set-piece-hashes" >
< h1 > set_piece_hashes()< / h1 >
< blockquote >
< pre class = "literal-block" >
template < class Fun>
void set_piece_hashes(create_torrent& t, boost::filesystem::path const& p, Fun f);
2008-12-02 09:41:02 +01:00
template < class Fun>
void set_piece_hashes(create_torrent& t, boost::filesystem::wpath const& p, Fun f);
2009-03-21 05:36:46 +01:00
template < class Fun>
void set_piece_hashes(create_torrent& t, boost::filesystem::path const& p, Fun f
, error_code& ec);
template < class Fun>
void set_piece_hashes(create_torrent& t, boost::filesystem::wpath const& p, Fun f
, error_code& ec);
2008-11-19 19:32:36 +01:00
void set_piece_hashes(create_torrent& t, boost::filesystem::path const& p);
2008-12-02 09:41:02 +01:00
void set_piece_hashes(create_torrent& t, boost::filesystem::wpath const& p);
2009-03-21 05:36:46 +01:00
void set_piece_hashes(create_torrent& t, boost::filesystem::path const& p
, error_code& ec);
void set_piece_hashes(create_torrent& t, boost::filesystem::wpath const& p
, error_code& ec);
2008-11-19 19:32:36 +01:00
< / pre >
< / blockquote >
< p > This function will assume that the files added to the torrent file exists at path
< tt class = "docutils literal" > < span class = "pre" > p< / span > < / tt > , read those files and hash the content and set the hashes in the < tt class = "docutils literal" > < span class = "pre" > create_torrent< / span > < / tt >
object. The optional function < tt class = "docutils literal" > < span class = "pre" > f< / span > < / tt > is called in between every hash that is set. < tt class = "docutils literal" > < span class = "pre" > f< / span > < / tt >
must have the following signature:< / p >
< pre class = "literal-block" >
void Fun(int);
< / pre >
2009-03-21 05:36:46 +01:00
< p > The overloads that don't take an < tt class = "docutils literal" > < span class = "pre" > error_code& < / span > < / tt > may throw an exception in case of a
file error, the other overloads sets the error code to reflect the error, if any.< / p >
2008-11-19 19:32:36 +01:00
< / div >
< div class = "section" id = "file-storage" >
< h1 > file_storage< / h1 >
< p > The < tt class = "docutils literal" > < span class = "pre" > file_storage< / span > < / tt > class represents a file list and the piece
size. Everything necessary to interpret a regular bittorrent storage
file structure. Its synopsis:< / p >
< pre class = "literal-block" >
class file_storage
{
public:
bool is_valid() const;
2009-01-13 22:49:03 +01:00
enum flags_t
{
pad_file = 1,
attribute_hidden = 2,
attribute_executable = 4
};
2008-11-19 19:32:36 +01:00
void add_file(file_entry const& e);
2009-01-13 22:49:03 +01:00
void add_file(fs::path const& p, size_type size, int flags = 0);
void add_file(fs::wpath const& p, size_type size, int flags = 0);
2008-11-19 19:32:36 +01:00
void rename_file(int index, std::string const& new_filename);
2008-12-02 09:41:02 +01:00
void rename_file(int index, std::wstring const& new_filename);
2008-11-19 19:32:36 +01:00
std::vector< file_slice> map_block(int piece, size_type offset
, int size) const;
peer_request map_file(int file, size_type offset, int size) const;
typedef std::vector< file_entry> ::const_iterator iterator;
typedef std::vector< file_entry> ::const_reverse_iterator reverse_iterator;
iterator begin() const;
iterator end() const;
reverse_iterator rbegin();
reverse_iterator rend() const;
int num_files() const;
file_entry const& at(int index) const;
size_type total_size() const;
void set_num_pieces(int n);
int num_pieces() const;
void set_piece_length(int l);
int piece_length() const;
int piece_size(int index) const;
void set_name(std::string const& n);
2008-12-02 09:41:02 +01:00
void set_name(std::wstring const& n);
2008-11-19 19:32:36 +01:00
const std::string& name() const;
void swap(file_storage& ti);
}
< / pre >
2009-01-13 22:49:03 +01:00
< div class = "section" id = "add-file" >
< h2 > add_file()< / h2 >
< blockquote >
< pre class = "literal-block" >
void add_file(file_entry const& e);
void add_file(fs::path const& p, size_type size, int flags = 0);
void add_file(fs::wpath const& p, size_type size, int flags = 0);
< / pre >
< / blockquote >
< p > Adds a file to the file storage. The < tt class = "docutils literal" > < span class = "pre" > flags< / span > < / tt > argument sets attributes on the file.
The file attributes is an extension and may not work in all bittorrent clients.
The possible arreibutes are:< / p >
< pre class = "literal-block" >
pad_file
attribute_hidden
attribute_executable
< / pre >
< / div >
2009-11-11 08:03:50 +01:00
< div class = "section" id = "id1" >
< h2 > add_file< / h2 >
< blockquote >
< pre class = "literal-block" >
void add_file(file_entry const& e);
void add_file(fs::path const& p, size_type size);
< / pre >
< / blockquote >
< p > Adds a file to the file storage. If more files than one are added,
certain restrictions to their paths apply. In a multi-file file
storage (torrent), all files must share the same root directory.< / p >
< p > That is, the first path element of all files must be the same.
This shared path element is also set to the name of the torrent. It
can be changed by calling < tt class = "docutils literal" > < span class = "pre" > set_name< / span > < / tt > .< / p >
< p > The built in functions to traverse a directory to add files will
make sure this requirement is fulfilled.< / p >
< / div >
2008-11-19 19:32:36 +01:00
< / div >
< div class = "section" id = "create-torrent" >
< h1 > create_torrent< / h1 >
< p > The < tt class = "docutils literal" > < span class = "pre" > create_torrent< / span > < / tt > class has the following synopsis:< / p >
< pre class = "literal-block" >
struct create_torrent
{
2010-03-20 09:26:44 +01:00
enum {
optimize = 1
, merkle = 2
, modification_time = 4
, symlink = 8
};
2009-03-21 05:36:46 +01:00
create_torrent(file_storage& fs, int piece_size = 0, int pad_size_limit = -1, int flags = optimize);
2008-11-19 19:32:36 +01:00
create_torrent(torrent_info const& ti);
entry generate() const;
file_storage const& files() const;
void set_comment(char const* str);
void set_creator(char const* str);
void set_hash(int index, sha1_hash const& h);
void add_url_seed(std::string const& url);
void add_node(std::pair< std::string, int> const& node);
void add_tracker(std::string const& url, int tier = 0);
2009-01-13 22:49:03 +01:00
void set_priv(bool p);
2008-11-19 19:32:36 +01:00
int num_pieces() const;
int piece_length() const;
int piece_size(int i) const;
2009-01-13 22:49:03 +01:00
bool priv() const;
2008-11-19 19:32:36 +01:00
};
< / pre >
2009-11-11 08:03:50 +01:00
< div class = "section" id = "id2" >
2008-11-19 19:32:36 +01:00
< h2 > create_torrent()< / h2 >
< blockquote >
< pre class = "literal-block" >
2010-03-20 09:26:44 +01:00
enum {
optimize = 1
, merkle = 2
, modification_time = 4
, symlink = 8
};
2009-03-21 05:36:46 +01:00
create_torrent(file_storage& fs, int piece_size = 0, int pad_size_limit = -1, int flags = optimize);
2008-11-19 19:32:36 +01:00
create_torrent(torrent_info const& ti);
< / pre >
< / blockquote >
2009-01-13 22:49:03 +01:00
< p > The < tt class = "docutils literal" > < span class = "pre" > piece_size< / span > < / tt > is the size of each piece in bytes. It must
be a multiple of 16 kiB. If a piece size of 0 is specified, a
2009-06-26 21:03:28 +02:00
piece_size will be calculated such that the torrent file is roughly 40 kB.< / p >
2009-01-13 22:49:03 +01:00
< p > If a < tt class = "docutils literal" > < span class = "pre" > pad_size_limit< / span > < / tt > is specified (other than -1), any file larger than
the specified number of bytes will be preceeded by a pad file to align it
2009-06-26 21:03:28 +02:00
with the start of a piece. The pad_file_limit is ignored unless the
2009-03-21 05:36:46 +01:00
< tt class = "docutils literal" > < span class = "pre" > optimize< / span > < / tt > flag is passed.< / p >
< p > The overload that takes a < tt class = "docutils literal" > < span class = "pre" > torrent_info< / span > < / tt > object will make a verbatim
2008-11-19 19:32:36 +01:00
copy of its info dictionary (to preserve the info-hash). The copy of
the info dictionary will be used by < tt class = "docutils literal" > < span class = "pre" > generate()< / span > < / tt > . This means
that none of the member functions of create_torrent that affects
2009-01-13 22:49:03 +01:00
the content of the info dictionary (such as < tt class = "docutils literal" > < span class = "pre" > set_hash()< / span > < / tt > ), will
2008-11-19 19:32:36 +01:00
have any affect.< / p >
2009-03-21 05:36:46 +01:00
< p > The < tt class = "docutils literal" > < span class = "pre" > flags< / span > < / tt > arguments specifies options for the torrent creation. It can
be any combination of the following flags:< / p >
< dl class = "docutils" >
< dt > optimize< / dt >
< dd > This will insert pad files to align the files to piece boundaries, for
optimized disk-I/O.< / dd >
< dt > merkle< / dt >
< dd > This will create a merkle hash tree torrent. A merkle torrent cannot
be opened in clients that don't specifically support merkle torrents.
The benefit is that the resulting torrent file will be much smaller and
not grow with more pieces. When this option is specified, it is
2009-06-26 21:03:28 +02:00
recommended to have a fairly small piece size, say 64 kiB.< / dd >
2010-03-20 09:26:44 +01:00
< dt > modification_time< / dt >
< dd > This will include the file modification time as part of the torrent.
This is not enabled by default, as it might cause problems when you
create a torrent from separate files with the same content, hoping to
yield the same info-hash. If the files have different modification times,
with this option enabled, you would get different info-hashes for the
files.< / dd >
< dt > symlink< / dt >
< dd > If this flag is defined, files that are symlinks get a symlink attribute
set on them. The file data will still be the same, the symlink will always
be followed when opening the file, but the file list will include the path
of the symlink so that the original directory structure can be reproduced
on the downloading side.< / dd >
2009-03-21 05:36:46 +01:00
< / dl >
2008-11-19 19:32:36 +01:00
< / div >
< div class = "section" id = "generate" >
< h2 > generate()< / h2 >
< blockquote >
< pre class = "literal-block" >
entry generate() const;
< / pre >
< / blockquote >
< p > This function will generate the .torrent file as a bencode tree. In order to
generate the flat file, use the bencode() function.< / p >
< p > It may be useful to add custom entries to the torrent file before bencoding it
and saving it to disk.< / p >
2009-11-11 08:03:50 +01:00
< p > If anything goes wrong during torrent generation, this function will return
an empty < tt class = "docutils literal" > < span class = "pre" > entry< / span > < / tt > structure. You can test for this condition by querying the
type of the entry:< / p >
< pre class = "literal-block" >
file_storage fs;
// add file ...
create_torrent t(fs);
// add trackers and piece hashes ...
e = t.generate();
if (e.type() == entry::undefined_t)
{
// something went wrong
}
< / pre >
< p > For instance, you cannot generate a torrent with 0 files in it. If you don't add
any files to the < tt class = "docutils literal" > < span class = "pre" > file_storage< / span > < / tt > , torrent generation will fail.< / p >
2008-11-19 19:32:36 +01:00
< / div >
2009-01-13 22:49:03 +01:00
< div class = "section" id = "set-comment" >
< h2 > set_comment()< / h2 >
< blockquote >
< pre class = "literal-block" >
void set_comment(char const* str);
< / pre >
< / blockquote >
< p > Sets the comment for the torrent. The string < tt class = "docutils literal" > < span class = "pre" > str< / span > < / tt > should be utf-8 encoded.
The comment in a torrent file is optional.< / p >
< / div >
< div class = "section" id = "set-creator" >
< h2 > set_creator()< / h2 >
< blockquote >
< pre class = "literal-block" >
void set_creator(char const* str);
< / pre >
< / blockquote >
< p > Sets the creator of the torrent. The string < tt class = "docutils literal" > < span class = "pre" > str< / span > < / tt > should be utf-8 encoded.
This is optional.< / p >
< / div >
< div class = "section" id = "set-hash" >
< h2 > set_hash()< / h2 >
< blockquote >
< pre class = "literal-block" >
void set_hash(int index, sha1_hash const& h);
< / pre >
< / blockquote >
< p > This sets the SHA-1 hash for the specified piece (< tt class = "docutils literal" > < span class = "pre" > index< / span > < / tt > ). You are required
to set the hash for every piece in the torrent before generating it. If you have
the files on disk, you can use the high level convenience function to do this.
2009-03-21 05:36:46 +01:00
See < a class = "reference internal" href = "#set-piece-hashes" > set_piece_hashes()< / a > .< / p >
2009-01-13 22:49:03 +01:00
< / div >
< div class = "section" id = "add-url-seed" >
< h2 > add_url_seed()< / h2 >
< blockquote >
< pre class = "literal-block" >
void add_url_seed(std::string const& url);
< / pre >
< / blockquote >
< p > This adds a url seed to the torrent. You can have any number of url seeds. For a
single file torrent, this should be an HTTP url, pointing to a file with identical
content as the file of the torrent. For a multi-file torrent, it should point to
a directory containing a directory with the same name as this torrent, and all the
files of the torrent in it.< / p >
< / div >
< div class = "section" id = "add-node" >
< h2 > add_node()< / h2 >
< blockquote >
< pre class = "literal-block" >
void add_node(std::pair< std::string, int> const& node);
< / pre >
< / blockquote >
< p > This adds a DHT node to the torrent. This especially useful if you're creating a
tracker less torrent. It can be used by clients to bootstrap their DHT node from.
The node is a hostname and a port number where there is a DHT node running.
You can have any number of DHT nodes in a torrent.< / p >
< / div >
< div class = "section" id = "add-tracker" >
< h2 > add_tracker()< / h2 >
< blockquote >
< pre class = "literal-block" >
void add_tracker(std::string const& url, int tier = 0);
< / pre >
< / blockquote >
< p > Adds a tracker to the torrent. This is not strictly required, but most torrents
2009-03-21 05:36:46 +01:00
use a tracker as their main source of peers. The url should be an < a class = "reference external" href = "http://" > http://< / a > or udp://
2009-01-13 22:49:03 +01:00
url to a machine running a bittorrent tracker that accepts announces for this torrent's
info-hash. The tier is the fallback priority of the tracker. All trackers with tier 0 are
tried first (in any order). If all fail, trackers with tier 1 are tried. If all of those
fail, trackers with tier 2 are tried, and so on.< / p >
< / div >
< div class = "section" id = "set-priv-priv" >
< h2 > set_priv() priv()< / h2 >
< blockquote >
< pre class = "literal-block" >
void set_priv(bool p);
bool priv() const;
< / pre >
< / blockquote >
< p > Sets and queries the private flag of the torrent.< / p >
< / div >
2008-11-19 19:32:36 +01:00
< / div >
< / div >
< div id = "footer" >
< span > Copyright © 2005 Rasterbar Software.< / span >
< / div >
< / div >
< script src = "http://www.google-analytics.com/urchin.js" type = "text/javascript" >
< / script >
< script type = "text/javascript" >
_uacct = "UA-1599045-1";
urchinTracker();
< / script >
< / div >
< / body >
< / html >