 
    
    | Version: | 2.0.10 | 
|---|
Table of contents
The interface of libtorrent consists of a few classes. The main class is the session, it contains the main loop that serves all torrents.
The basic usage is as follows:
construct a session, possibly passing in the state from a previous session. use read_session_params() and pass in the resulting session_params object to the session constructor.
start extensions (see add_extension()).
start DHT, LSD, UPnP, NAT-PMP etc (see start_dht(), start_lsd(), start_upnp() and start_natpmp()).
parse .torrent-files and add them to the session (see torrent_info, async_add_torrent() and add_torrent())
main loop (see session)
- poll for alerts (see wait_for_alert(), pop_alerts())
- handle updates to torrents, (see state_update_alert).
- handle other alerts, (see alert).
- query the session for information (see session::status()).
- add and remove torrents from the session (remove_torrent())
save resume data for all torrent_handles (optional, see save_resume_data())
save session state (see session_state() and write_session_params())
destruct session object
Each class and function is described in this manual, you may want to have a look at the tutorial as well.
For a description on how to create torrent files, see create_torrent.
Forward declaring types from the libtorrent namespace is discouraged as it may break in future releases. Instead include libtorrent/fwd.hpp for forward declarations of all public types in libtorrent.
A common problem developers are facing is torrents stopping without explanation. Here is a description on which conditions libtorrent will stop your torrents, how to find out about it and what to do about it.
Make sure to keep track of the paused state, the error state and the upload mode of your torrents. By default, torrents are auto-managed, which means libtorrent will pause, resume, scrape them and take them out of upload-mode automatically.
Whenever a torrent encounters a fatal error, it will be stopped, and the torrent_status::error will describe the error that caused it. If a torrent is auto managed, it is scraped periodically and paused or resumed based on the number of downloaders per seed. This will effectively seed torrents that are in the greatest need of seeds.
If a torrent hits a disk write error, it will be put into upload mode. This means it will not download anything, but only upload. The assumption is that the write error is caused by a full disk or write permission errors. If the torrent is auto-managed, it will periodically be taken out of the upload mode, trying to write things to the disk again. This means torrent will recover from certain disk errors if the problem is resolved. If the torrent is not auto managed, you have to call set_upload_mode() to turn downloading back on again.
For a more detailed guide on how to trouble shoot performance issues, see troubleshooting
libtorrent maintains a stable ABI for versions with the same major and minor versions.
e.g. libtorrent-1.2.0 is ABI compatible with libtorrent-1.2.1 but not with libtorrent-1.1
There are a few typedefs in the libtorrent namespace which pulls in network types from the boost::asio namespace. These are:
using address = boost::asio::ip::address; using address_v4 = boost::asio::ip::address_v4; using address_v6 = boost::asio::ip::address_v6; using boost::asio::ip::tcp; using boost::asio::ip::udp;
These are declared in the <libtorrent/socket.hpp> header.
The using statements will give easy access to:
tcp::endpoint udp::endpoint
Which are the endpoint types used in libtorrent. An endpoint is an address with an associated port.
For documentation on these types, please refer to the asio documentation.
Many functions in libtorrent have two versions, one that throws exceptions on errors and one that takes an error_code reference which is filled with the error code on errors.
On exceptions, libtorrent will throw boost::system::system_error exceptions carrying an error_code describing the underlying error.
The error_code::message() function will typically return a localized error string, for system errors. That is, errors that belong to the generic or system category.
Errors that belong to the libtorrent error category are not localized however, they are only available in English. In order to translate libtorrent errors, compare the error category of the error_code object against lt::libtorrent_category(), and if matches, you know the error code refers to the list above. You can provide your own mapping from error code to string, which is localized. In this case, you cannot rely on error_code::message() to generate your strings.
The numeric values of the errors are part of the API and will stay the same, although new error codes may be appended at the end.
Here's a simple example of how to translate error codes:
std::string error_code_to_string(boost::system::error_code const& ec) { if (ec.category() != lt::libtorrent_category()) { return ec.message(); } // the error is a libtorrent error int code = ec.value(); static const char const* swedish[] = { "inget fel", "en fil i torrenten kolliderar med en fil fran en annan torrent", "hash check misslyckades", "torrentfilen ar inte en dictionary", "'info'-nyckeln saknas eller ar korrupt i torrentfilen", "'info'-faltet ar inte en dictionary", "'piece length' faltet saknas eller ar korrupt i torrentfilen", "torrentfilen saknar namnfaltet", "ogiltigt namn i torrentfilen (kan vara en attack)", // ... more strings here }; // use the default error string in case we don't have it // in our translated list if (code < 0 || code >= sizeof(swedish)/sizeof(swedish[0])) return ec.message(); return swedish[code]; }
Magnet links are URIs that includes an info-hash, a display name and optionally a tracker url. The idea behind magnet links is that an end user can click on a link in a browser and have it handled by a bittorrent application, to start a download, without any .torrent file.
The format of the magnet URI is:
magnet:?xt=urn:btih: Base16 encoded info-hash [ &dn= name of download ] [ &tr= tracker URL ]*
In order to download just the metadata (.torrent file) from a magnet link, set the torrent_flags::upload_mode flag in add_torrent_params before adding the it.
In this case, when the metadata is received from the swarm, the torrent will still be running, but it will disconnect the majority of peers (since connections to peers that already have the metadata are redundant). It will keep seeding the metadata only.
Note that this doesn't prevent empty files from being created, if the torrent contains any. If you need to prevent that, you can either set file_priority to a long list of zeros (since the number of files is not known in advance), or set save_path to an invalid path.
To save a .torrent file from a torrent that was added by magnet link (or added any way really):
The resume data format is very similar to the .torrent file format, and when including the info-dict in the resume data, the resume file can be used as a .torrent file (with just a few minor exceptions).
BitTorrent v2 introduces a number of features outlined in this blog post as well as BEP 52. The v2 protocol introduces the possibility to use a merkle hash tree instead of a flat list of piece hashes. It also supports hybrid torrents, that are both valid classing torrents (v1) as well as valid v2 torrents. Hybrid torrents contain both a flat list of piece hashes as well as a merkle hash tree.
This introduces a few new error cases. A hybrid torrent may have mismatching v1 and v2 hashes. Since v2 torrents use SHA-256 and v1 uses SHA-1 the fact that the hashes are mismatching won't be detectable until the piece has been downloaded. It results in a torrent_inconsistent_hashes error.
A magnet link may contain just a v1 info-hash or a v2 info-hash. If two separate magnet links, one v1-only and one v2-only, end up resolving to the same hybrid torrent, both torrent_handle objects are put into an error state of duplicate_torrent. In this state, one of them has to be removed, and the other one can be resumed, in order to download the metadata again.
When a conflict between two torrents occur, a torrent_conflict_alert is posted. This alert derives from torrent_alert, so is associated with a torrent_handle. It contains a second torrent_handle referring to the other torrent in the conflict as well as the metadata that was downloaded. One way to resolve the conflict is to remove both torrents and add it back using the metadata supplied in the torrent_conflict_alert.
When a v1-only or v2-only magnet link resolves to a hybrid torrent, the info_hash_t object associated with the torrent will be updated to include both the v1 and v2 info hash. This applies both to torrent_handle::info_hashes() as well as torrent_info::info_hashes().
libtorrent supports queuing. Queuing is a mechanism to automatically pause and resume torrents based on certain criteria. The criteria depends on the overall state the torrent is in (checking, downloading or seeding).
To opt-out of the queuing logic, make sure your torrents are added with the torrent_flags::auto_managed bit cleared from add_torrent_params::flags. Or call torrent_handle::unset_flags() and pass in torrent_flags::auto_managed on the torrent handle.
The overall purpose of the queuing logic is to improve performance under arbitrary torrent downloading and seeding load. For example, if you want to download 100 torrents on a limited home connection, you improve performance by downloading them one at a time (or maybe two at a time), over downloading them all in parallel. The benefits are:
There are fundamentally 3 separate queues:
Every torrent that is not seeding has a queue number associated with it, this is its place in line to be started. See torrent_status::queue_position.
On top of the limits of each queue, there is an over arching limit, set in settings_pack::active_limit. The auto manager will never start more than this number of torrents (with one exception described below). Non-auto-managed torrents are exempt from this logic, and not counted.
At a regular interval, torrents are checked if there needs to be any re-ordering of which torrents are active and which are queued. This interval can be controlled via settings_pack::auto_manage_interval.
For queuing to work, resume data needs to be saved and restored for all torrents. See torrent_handle::save_resume_data().
The torrents in the front of the queue are started and the rest are ordered by their queue position. Any newly added torrent is placed at the end of the queue. Once a torrent is removed or turns into a seed, its queue position is -1 and all torrents that used to be after it in the queue, decreases their position in order to fill the gap.
The queue positions are always contiguous, in a sequence without any gaps.
Lower queue position means closer to the front of the queue, and will be started sooner than torrents with higher queue positions.
To query a torrent for its position in the queue, or change its position, see: torrent_handle::queue_position(), torrent_handle::queue_position_up(), torrent_handle::queue_position_down(), torrent_handle::queue_position_top() and torrent_handle::queue_position_bottom().
The checking queue affects torrents in the torrent_status::checking or torrent_status::allocating state that are auto-managed.
The checking queue will make sure that (of the torrents in its queue) no more than settings_pack::active_checking_limit torrents are started at any given time. Once a torrent completes checking and moves into a different state, the next in line will be started for checking.
Any torrent added force-started or force-stopped (i.e. the auto managed flag is not set), will not be subject to this limit and they will all check independently and in parallel.
Once a torrent completes the checking of its files, or resume data, it will be put in the queue for downloading and potentially start downloading immediately. In order to add a torrent and check its files without starting the download, it can be added in stop_when_ready mode. See add_torrent_params::flag_stop_when_ready. This flag will stop the torrent once it is ready to start downloading.
This is conceptually the same as waiting for the torrent_checked_alert and then call:
h.set_flags(torrent_flags::paused, torrent_flags::paused | torrent_flags::auto_managed);
With the important distinction that it entirely avoids the brief window where the torrent is in downloading state.
Similarly to the checking queue, the downloading queue will make sure that no more than settings_pack::active_downloads torrents are in the downloading state at any given time.
The torrent_status::queue_position is used again here to determine who is next in line to be started once a downloading torrent completes or is stopped/removed.
The seeding queue does not use torrent_status::queue_position to determine which torrent to seed. Instead, it estimates the demand for the torrent to be seeded. A torrent with few other seeds and many downloaders is assumed to have a higher demand of more seeds than one with many seeds and few downloaders.
It limits the number of started seeds to settings_pack::active_seeds.
On top of this basic bias, seed priority can be controller by specifying a seed ratio (the upload to download ratio), a seed-time ratio (the download time to seeding time ratio) and a seed-time (the absolute time to be seeding a torrent). Until all those targets are hit, the torrent will be prioritized for seeding.
Among torrents that have met their seed target, torrents where we don't know of any other seed take strict priority.
In order to avoid flapping, torrents that were started less than 30 minutes ago also have priority to keep seeding.
Finally, for torrents where none of the above apply, they are prioritized based on the download to seed ratio.
The relevant settings to control these limits are settings_pack::share_ratio_limit, settings_pack::seed_time_ratio_limit and settings_pack::seed_time_limit.
In addition to simply starting and stopping torrents, the queuing mechanism can have more fine grained control of the resources used by torrents.
In addition to the downloading and seeding limits, there are limits on actions torrents perform. The downloading and seeding limits control whether peers are allowed at all, and if peers are not allowed, torrents are stopped and don't do anything. If peers are allowed, torrents may:
Each of those actions are associated with a cost and hence may need a separate limit. These limits are controlled by settings_pack::active_tracker_limit, settings_pack::active_dht_limit and settings_pack::active_lsd_limit respectively.
Specifically, announcing to a tracker is typically cheaper than announcing to the DHT. settings_pack::active_dht_limit will limit the number of torrents that are allowed to announce to the DHT. The highest priority ones will, and the lower priority ones won't. The will still be considered started though, and any incoming peers will still be accepted.
If you do not wish to impose such limits (basically, if you do not wish to have half-started torrents) make sure to set these limits to -1 (infinite).
In the case where active_downloads + active_seeds > active_limit, there's an ambiguity whether the downloads should be satisfied first or the seeds. To disambiguate this case, the settings_pack::auto_manage_prefer_seeds determines whether seeds are preferred or not.
Torrents that are not transferring any bytes (downloading or uploading) have a relatively low cost to be started. It's possible to exempt such torrents from the download and seed queues by setting settings_pack::dont_count_slow_torrents to true.
Since it sometimes may take a few minutes for a newly started torrent to find peers and be unchoked, or find peers that are interested in requesting data, torrents are not considered inactive immediately. There must be an extended period of no transfers before it is considered inactive and exempt from the queuing limits.
The fast resume mechanism is a way to remember which pieces are downloaded and where they are put between sessions. You can generate fast resume data by:
When adding a torrent using resume data, load it using read_resume_data(). This populates an add_torrent_params object, which can be passed directly to add_torrent() or async_add_torrent() on the session object. libtorrent will not check the piece hashes then, and rely on the information given in the fast-resume data. The fast-resume data also contains information about which blocks, in the unfinished pieces, were downloaded, so it will not have to start from scratch on the partially downloaded pieces.
To use the fast-resume data you pass it to read_resume_data(), which will return an add_torrent_params object. Fields of this object can then be altered before passing it to async_add_torrent() or add_torrent(). The session will then skip the time consuming checks. It may have to do the checking anyway, if the fast-resume data is corrupt or doesn't fit the storage for that torrent.
The file format is a bencoded dictionary containing the following fields:
| file-format | string: "libtorrent resume file" | ||||||
| file-version | integer: 1 | ||||||
| info-hash | string, the info hash of the torrent this data is saved for. This is a 20 byte SHA-1 hash of the info section of the torrent if this is a v1 or v1+v2-hybrid torrent. | ||||||
| info-hash2 | string, the v2 info hash of the torrent this data is saved. for, in case it is a v2 or v1+v2-hybrid torrent. This is a 32 byte SHA-256 hash of the info section of the torrent. | ||||||
| pieces | A string with piece flags, one character per piece. Bit 1 means we have that piece. Bit 2 means we have verified that this piece is correct. This only applies when the torrent is in seed_mode. | ||||||
| total_uploaded | integer. The number of bytes that have been uploaded in total for this torrent. | ||||||
| total_downloaded | integer. The number of bytes that have been downloaded in total for this torrent. | ||||||
| active_time | integer. The number of seconds this torrent has been active. i.e. not paused. | ||||||
| seeding_time | integer. The number of seconds this torrent has been active and seeding. | ||||||
| last_upload | integer. The number of seconds since epoch when we last uploaded payload to a peer on this torrent. | ||||||
| last_download | integer. The number of seconds since epoch when we last downloaded payload from a peer on this torrent. | ||||||
| upload_rate_limit | integer. In case this torrent has a per-torrent upload rate limit, this is that limit. In bytes per second. | ||||||
| download_rate_limit | integer. The download rate limit for this torrent in case one is set, in bytes per second. | ||||||
| max_connections | integer. The max number of peer connections this torrent may have, if a limit is set. | ||||||
| max_uploads | integer. The max number of unchoked peers this torrent may have, if a limit is set. | ||||||
| file_priority | list of integers. One entry per file in the torrent. Each entry is the priority of the file with the same index. | ||||||
| piece_priority | string of bytes. Each byte is interpreted as an integer and is the priority of that piece. | ||||||
| seed_mode | integer. 1 if the torrent is in seed mode, 0 otherwise. | ||||||
| upload_mode | integer. 1 if the torrent_flags::upload_mode is set. | ||||||
| share_mode | integer. 1 if the torrent_flags::share_mode is set. | ||||||
| apply_ip_filter | integer. 1 if the torrent_flags::apply_ip_filter is set. | ||||||
| paused | integer. 1 if the torrent is paused, 0 otherwise. | ||||||
| auto_managed | integer. 1 if the torrent is auto managed, otherwise 0. | ||||||
| super_seeding | integer. 1 if the torrent_flags::super_seeding is set. | ||||||
| sequential_download | integer. 1 if the torrent is in sequential download mode, 0 otherwise. | ||||||
| stop_when_ready | integer. 1 if the torrent_flags::stop_when_ready is set. | ||||||
| disable_dht | integer. 1 if the torrent_flags::disable_dht is set. | ||||||
| disable_lsd | integer. 1 if the torrent_flags::disable_lsd is set. | ||||||
| disable_pex | integer. 1 if the torrent_flags::disable_pex is set. | ||||||
| trackers | list of lists of strings. The top level list lists all tracker tiers. Each second level list is one tier of trackers. | ||||||
| mapped_files | list of strings. If any file in the torrent has been renamed, this entry contains a list of all the filenames. In the same order as in the torrent file. | ||||||
| url-list | list of strings. List of url-seed URLs used by this torrent. The URLs are expected to be properly encoded and not contain any illegal url characters. | ||||||
| httpseeds | list of strings. List of HTTP seed URLs used by this torrent. The URLs are expected to be properly encoded and not contain any illegal url characters. | ||||||
| trees | list. In case this is a v2 (or v1+v2-hybrid) torrent, this is an optional list containing the merkle tree nodes we know of so far, for all files. It's a list of dictionaries, one entry for each file in the torrent. The entries have the following structure: 
 | ||||||
| save_path | string. The save path where this torrent was saved. This is especially useful when moving torrents with move_storage() since this will be updated. | ||||||
| peers | string. This string contains IPv4 and port pairs of peers we were connected to last session. The endpoints are in compact representation. 4 bytes IPv4 address followed by 2 bytes port. Hence, the length of this string should be divisible by 6. | ||||||
| banned_peers | string. This string has the same format as peers but instead represent IPv4 peers that we have banned. | ||||||
| peers6 | string. This string contains IPv6 and port pairs of peers we were connected to last session. The endpoints are in compact representation. 16 bytes IPv6 address followed by 2 bytes port. The length of this string should be divisible by 18. | ||||||
| banned_peers6 | string. This string has the same format as peers6 but instead represent IPv6 peers that we have banned. | ||||||
| info | If this field is present, it should be the info-dictionary of the torrent this resume data is for. Its SHA-1 hash must match the one in the info-hash field. When present, the torrent is loaded from here, meaning the torrent can be added purely from resume data (no need to load the .torrent file separately). This may have performance advantages. | ||||||
| unfinished | list of dictionaries. Each dictionary represents an piece, and has the following layout: 
 | ||||||
| allocation | The allocation mode for the storage. Can be either allocate or sparse. | 
There are two modes in which storage (files on disk) are allocated in libtorrent.
On filesystems that supports sparse files, this allocation mode will only use as much space as has been downloaded.
The main drawback of this mode is that it may create heavily fragmented files.
- It does not require an allocation pass on startup.
When a torrent is started in full allocation mode, the disk-io thread will make sure that the entire storage is allocated, and fill any gaps with zeros. It will of course still check for existing pieces and fast resume data. The main drawbacks of this mode are:
- It may take longer to start the torrent, since it will need to fill the files with zeros. This delay is linear to the size of the download.
- The download may occupy unnecessary disk space between download sessions.
- Disk caches usually perform poorly with random access to large files and may slow down the download some.
The benefits of this mode are:
- Downloaded pieces are written directly to their final place in the files and the total number of disk operations will be fewer and may also play nicer to the filesystem file allocation, and reduce fragmentation.
- No risk of a download failing because of a full disk during download, once all files have been created.
There are two kinds of HTTP seeding. One with that assumes a smart (and polite) client and one that assumes a smart server. These are specified in BEP 19 and BEP 17 respectively.
libtorrent supports both. In the libtorrent source code and API, BEP 19 URLs are typically referred to as url seeds and BEP 17 URLs are typically referred to as HTTP seeds.
The libtorrent implementation of BEP 19 assumes that, if the URL ends with a slash ('/'), the filename should be appended to it in order to request pieces from that file. The way this works is that if the torrent is a single-file torrent, only that filename is appended. If the torrent is a multi-file torrent, the torrent's name '/' the file name is appended. This is the same directory structure that libtorrent will download torrents into.
There is limited support for HTTP redirects. In case some files are redirected to different hosts, the files must be piece aligned or padded to be piece aligned.
The piece picker in libtorrent has the following features:
It is optimized by, at all times, keeping a list of pieces ordered by rarity, randomly shuffled within each rarity class. This list is organized as a single vector of contiguous memory in RAM, for optimal memory locality and to eliminate heap allocations and frees when updating rarity of pieces.
Expensive events, like a peer joining or leaving, are evaluated lazily, since it's cheaper to rebuild the whole list rather than updating every single piece in it. This means as long as no blocks are picked, peers joining and leaving is no more costly than a single peer joining or leaving. Of course the special cases of peers that have all or no pieces are optimized to not require rebuilding the list.
The normal mode of the picker is of course rarest first, meaning pieces that few peers have are preferred to be downloaded over pieces that more peers have. This is a fundamental algorithm that is the basis of the performance of bittorrent. However, the user may set the piece picker into sequential download mode. This mode simply picks pieces sequentially, always preferring lower piece indices.
When a torrent starts out, picking the rarest pieces means increased risk that pieces won't be completed early (since there are only a few peers they can be downloaded from), leading to a delay of having any piece to offer to other peers. This lack of pieces to trade, delays the client from getting started into the normal tit-for-tat mode of bittorrent, and will result in a long ramp-up time. The heuristic to mitigate this problem is to, for the first few pieces, pick random pieces rather than rare pieces. The threshold for when to leave this initial picker mode is determined by settings_pack::initial_picker_threshold.
An orthogonal setting is reverse order, which is used for snubbed peers. Snubbed peers are peers that appear very slow, and might have timed out a piece request. The idea behind this is to make all snubbed peers more likely to be able to do download blocks from the same piece, concentrating slow peers on as few pieces as possible. The reverse order means that the most common pieces are picked, instead of the rarest pieces (or in the case of sequential download, the last pieces, instead of the first).
Peers that have participated in a piece that failed the hash check, may be put in parole mode. This means we prefer downloading a full piece from this peer, in order to distinguish which peer is sending corrupt data. Whether to do this is or not is controlled by settings_pack::use_parole_mode.
In parole mode, the piece picker prefers picking one whole piece at a time for a given peer, avoiding picking any blocks from a piece any other peer has contributed to (since that would defeat the purpose of parole mode).
This setting determines if partially downloaded or requested pieces should always be preferred over other pieces. The benefit of doing this is that the number of partial pieces is minimized (and hence the turn-around time for downloading a block until it can be uploaded to others is minimized). It also puts less stress on the disk cache, since fewer partial pieces need to be kept in the cache. Whether or not to enable this is controlled by setting_pack::prioritize_partial_pieces.
The main benefit of not prioritizing partial pieces is that the rarest first algorithm gets to have more influence on which pieces are picked. The picker is more likely to truly pick the rarest piece, and hence improving the performance of the swarm.
This setting is turned on automatically whenever the number of partial pieces in the piece picker exceeds the number of peers we're connected to times 1.5. This is in order to keep the waste of partial pieces to a minimum, but still prefer rarest pieces.
The prefer whole pieces setting makes the piece picker prefer picking entire pieces at a time. This is used by web connections (both http seeding standards), in order to be able to coalesce the small bittorrent requests to larger HTTP requests. This significantly improves performance when downloading over HTTP.
It is also used by peers that are downloading faster than a certain threshold. The main advantage is that these peers will better utilize the other peer's disk cache, by requesting all blocks in a single piece, from the same peer.
This threshold is controlled by the settings_pack::whole_pieces_threshold setting.
TODO: piece priorities
The settings_pack::listen_interfaces setting is used to specify which interfaces/IP addresses to listen on, and accept incoming connections via.
Each item in listen_interfaces is an IP address or a device name, followed by a listen port number. Each item (called listen_socket_t) will have the following objects associated with it:
A listen_socket_t item may be specified to only be a local network (with the l suffix). Such listen socket will only be used to talk to peers and trackers within the same local network. The netmask defining the network is queried from the operating system by enumerating network interfaces.
An item that's considered to be "local network" will not be used to announce to trackers outside of that network. For example, 10.0.0.2:6881l is marked as "local network" and it will only be used as the source address announcing to a tracker if the tracker is also within the same local network (e.g. 10.0.0.0/8).
The NAT-PMP/PCP and UPnP port mapper objects are only created for networks that are expected to be externally available (i.e. not "local network"). If there are multiple subnets connected to the internet, they will have separate port mappings.
If a device name is specified, libtorrent will expand it to the IP addresses associated with that device, but also retain the device name in order to attempt to bind the listen sockets to that specific device.
If an IP address is the unspecified address (i.e. 0.0.0.0 or ::), libtorrent will expand it to specific IP addresses. This expansion will enumerate all addresses it can find for the corresponding address family. The expanded IP addresses are considered "local network" if any of the following conditions are met:
A listen_socket_t item is considered able to route to a destination address if any of these hold:
The ability to route to an address is used when determining whether to announce to a tracker from a listen_socket_t and whether to open a SOCKS5 UDP tunnel for a listen_socket_t.
Note that the actual IP stack routing table is not considered for this purpose. This mechanism is to determine which IP addresses should be announced to trackers.
Trackers are announced to from all network interfaces listening for incoming connections. However, interfaces that cannot be used to reach the tracker, such as loopback, are not used as the source address for announces. A listen_socket_t item that can route to at least one of the tracker IP addresses will be used as the source address for an announce. Each such item will also have an announce_endpoint item associated with it, in the tracker list.
If a tracker can be reached on a loopback address, then the loopback interface will be used to announce to that tracker. But under normal circumstances, loopback will not be used for announcing to trackers.
For more details, see BEP 7.
When using a SOCKS5 proxy, each interface that can route to one of the SOCKS5 proxy's addresses will be used to open a UDP tunnel, via that proxy. For example, if a client has both IPv4 and IPv6 connectivity, but the socks5 proxy only resolves to IPv4, only the IPv4 address will have a UDP tunnel. In that case, the IPv6 connection will not be used, since it cannot use the proxy.
libtorrent supports a choking algorithm that automatically determines the number of upload slots (unchoke slots) based on the upload rate to peers. It is controlled by the settings_pack::choking_algorithm setting. The settings_pack::unchoke_slots_limit is ignored in this mode.
The algorithm is designed to stay stable, and not oscillate the number of upload slots.
The initial rate threshold is set to settings_pack::rate_choker_initial_threshold.
It sorts all peers by on the rate at which we are uploading to them.
In other words, the more upload slots you have, the higher rate does the slowest unchoked peer upload at in order to open another slot.
In order to improve performance, libtorrent supports a feature called predictive piece announce. When enabled, it will make libtorrent announce that we have pieces to peers, before we truly have them. The most important case is to announce a piece as soon as it has been downloaded and passed the hash check, but not yet been written to disk. In this case, there is a risk the piece will fail to be written to disk, in which case we won't have the piece anymore, even though we announced it to peers.
The other case is when we're very close to completing the download of a piece and assume it will pass the hash check, we can announce it to peers to make it available one round-trip sooner than otherwise. This lets libtorrent start uploading the piece to interested peers immediately when the piece complete, instead of waiting one round-trip for the peers to request it.
This makes for the implementation slightly more complicated, since piece will have more states and more complicated transitions. For instance, a piece could be:
Once a piece is fully downloaded, the hash check could complete before any of the write operations or it could complete after all write operations are complete.
The peer classes feature in libtorrent allows a client to define custom groups of peers and rate limit them individually. Each such group is called a peer class. There are a few default peer classes that are always created:
The TCP peers class is used by the uTP/TCP balancing logic, if it's enabled, to throttle TCP peers. The global and local classes are used to adjust the global rate limits.
When the rate limits are adjusted for a specific torrent, a class is created implicitly for that torrent.
The default peer class IDs are defined as enums in the session class:
enum { global_peer_class_id, tcp_peer_class_id, local_peer_class_id };
The default peer classes are automatically created on session startup, and configured to apply to each respective type of connection. There's nothing preventing a client from reconfiguring the peer class ip- and type filters to disable or customize which peers they apply to. See set_peer_class_filter() and set_peer_class_type_filter().
A peer class can be considered a more general form of labels that some clients have. Peer classes however are not just applied to torrents, but ultimately the peers.
Peer classes can be created with the create_peer_class() call (on the session object), and deleted with the delete_peer_class() call.
Peer classes are configured with the set_peer_class() get_peer_class() calls.
Custom peer classes can be assigned based on the peer's IP address or the type of transport protocol used. See set_peer_class_filter() and set_peer_class_type_filter() for more information.
Here are a few examples of common peer class operations.
To make the global rate limit apply to local peers as well, update the IP-filter based peer class assignment:
std::uint32_t const mask = 1 << lt::session::global_peer_class_id; ip_filter f; // for every IPv4 address, assign the global peer class f.add_rule(make_address("0.0.0.0"), make_address("255.255.255.255"), mask); // for every IPv6 address, assign the global peer class f.add_rule(make_address("::") , make_address("ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff") , mask); ses.set_peer_class_filter(f);
To make uTP sockets exempt from rate limiting:
peer_class_type_filter flt = ses.get_peer_class_type_filter(); // filter out the global and local peer class for uTP sockets, if these // classes are set by the IP filter flt.disallow(peer_class_type_filter::utp_socket, session::global_peer_class_id); flt.disallow(peer_class_type_filter::utp_socket, session::local_peer_class_id); // this filter should not add the global or local peer class to utp sockets flt.remove(peer_class_type_filter::utp_socket, session::global_peer_class_id); flt.remove(peer_class_type_filter::utp_socket, session::local_peer_class_id); ses.set_peer_class_type_filter(flt);
To make all peers on the internal network not subject to throttling:
std::uint32_t const mask = 1 << lt::session::global_peer_class_id; ip_filter f; // for every IPv4 address, assign the global peer class f.add_rule(make_address("0.0.0.0"), make_address("255.255.255.255"), mask); // for every address on the local metwork, set the mask to 0 f.add_rule(make_address("10.0.0.0"), make_address("10.255.255.255"), 0); ses.set_peer_class_filter(f);
Torrents may have an SSL root (CA) certificate embedded in them. Such torrents are called SSL torrents. An SSL torrent talks to all bittorrent peers over SSL. The protocols are layered like this:
 
During the SSL handshake, both peers need to authenticate by providing a certificate that is signed by the CA certificate found in the .torrent file. These peer certificates are expected to be provided to peers through some other means than bittorrent. Typically by a peer generating a certificate request which is sent to the publisher of the torrent, and the publisher returning a signed certificate.
In libtorrent, set_ssl_certificate() in torrent_handle is used to tell libtorrent where to find the peer certificate and the private key for it. When an SSL torrent is loaded, the torrent_need_cert_alert is posted to remind the user to provide a certificate.
A peer connecting to an SSL torrent MUST provide the SNI TLS extension (server name indication). The server name is the hex encoded info-hash of the torrent to connect to. This is required for the client accepting the connection to know which certificate to present.
SSL connections are accepted on a separate socket from normal bittorrent connections. To enable support for SSL torrents, add a listen interface to the settings_pack::listen_interfaces setting with the s suffix. For example:
0.0.0.0:6881,0.0.0.0:6882s
That will listen for normal bittorrent connections on port 6881 and for SSL torrent connections on port 6882.
This feature is only available if libtorrent is built with SSL torrent support (TORRENT_SSL_PEERS) and requires at least OpenSSL version 1.0, since it needs SNI support.
Peer certificates must have at least one SubjectAltName field of type DNSName. At least one of the fields must exactly match the name of the torrent. This is a byte-by-byte comparison, the UTF-8 encoding must be identical (i.e. there's no unicode normalization going on). This is the recommended way of verifying certificates for HTTPS servers according to RFC 2818. Note the difference that for torrents only DNSName fields are taken into account (not IP address fields). The most specific (i.e. last) Common Name field is also taken into account if no SubjectAltName did not match.
If any of these fields contain a single asterisk ("*"), the certificate is considered covering any torrent, allowing it to be reused for any torrent.
The purpose of matching the torrent name with the fields in the peer certificate is to allow a publisher to have a single root certificate for all torrents it distributes, and issue separate peer certificates for each torrent. A peer receiving a certificate will not necessarily be able to access all torrents published by this root certificate (only if it has a "star cert").
To test incoming SSL connections to an SSL torrent, one can use the following openssl command:
openssl s_client -cert <peer-certificate>.pem -key <peer-private-key>.pem -CAfile \ <torrent-cert>.pem -debug -connect 127.0.0.1:4433 -tls1 -servername <info-hash>
To create a root certificate, the Distinguished Name (DN) is not taken into account by bittorrent peers. You still need to specify something, but from libtorrent's point of view, it doesn't matter what it is. libtorrent only makes sure the peer certificates are signed by the correct root certificate.
One way to create the certificates is to use the CA.sh script that comes with openssl, like this (don't forget to enter a common Name for the certificate):
CA.sh -newca CA.sh -newreq CA.sh -sign
The torrent certificate is located in ./demoCA/private/demoCA/cacert.pem, this is the pem file to include in the .torrent file.
The peer's certificate is located in ./newcert.pem and the certificate's private key in ./newkey.pem.
libtorrent provides a mechanism to query performance and statistics counters from its internals.
The statistics consists of two fundamental types. counters and gauges. A counter is a monotonically increasing value, incremented every time some event occurs. For example, every time the network thread wakes up because a socket became readable will increment a counter. Another example is every time a socket receives n bytes, a counter is incremented by n.
Counters are the most flexible of metrics. It allows the program to sample the counter at any interval, and calculate average rates of increments to the counter. Some events may be rare and need to be sampled over a longer period in order to get useful rates, where other events may be more frequent and evenly distributed that sampling it frequently yields useful values. Counters also provides accurate overall counts. For example, converting samples of a download rate into a total transfer count is not accurate and takes more samples. Converting an increasing counter into a rate is easy and flexible.
Gauges measure the instantaneous state of some kind. This is used for metrics that are not counting events or flows, but states that can fluctuate. For example, the number of torrents that are currently being downloaded.
It's important to know whether a value is a counter or a gauge in order to interpret it correctly. In order to query libtorrent for which counters and gauges are available, call session_stats_metrics(). This will return metadata about the values available for inspection in libtorrent. It will include whether a value is a counter or a gauge. The key information it includes is the index used to extract the actual measurements for a specific counter or gauge.
In order to take a sample, call post_session_stats() in the session object. This will result in a session_stats_alert being posted. In this alert object, there is an array of values, these values make up the sample. The value index in the stats metric indicates which index the metric's value is stored in.
The mapping between metric and value is not stable across versions of libtorrent. Always query the metrics first, to find out the index at which the value is stored, before interpreting the values array in the session_stats_alert. The mapping will not change during the runtime of your process though, it's tied to a specific libtorrent version. You only have to query the mapping once on startup (or every time libtorrent.so is loaded, if it's done dynamically).
The available stats metrics are:
| name | type | 
|---|---|
| peer.error_peers | counter | 
| peer.disconnected_peers | counter | 
error_peers is the total number of peer disconnects caused by an error (not initiated by this client) and disconnected initiated by this client (disconnected_peers).
| name | type | 
|---|---|
| peer.eof_peers | counter | 
| peer.connreset_peers | counter | 
| peer.connrefused_peers | counter | 
| peer.connaborted_peers | counter | 
| peer.notconnected_peers | counter | 
| peer.perm_peers | counter | 
| peer.buffer_peers | counter | 
| peer.unreachable_peers | counter | 
| peer.broken_pipe_peers | counter | 
| peer.addrinuse_peers | counter | 
| peer.no_access_peers | counter | 
| peer.invalid_arg_peers | counter | 
| peer.aborted_peers | counter | 
these counters break down the peer errors into more specific categories. These errors are what the underlying transport reported (i.e. TCP or uTP)
| name | type | 
|---|---|
| peer.piece_requests | counter | 
| peer.max_piece_requests | counter | 
| peer.invalid_piece_requests | counter | 
| peer.choked_piece_requests | counter | 
| peer.cancelled_piece_requests | counter | 
| peer.piece_rejects | counter | 
the total number of incoming piece requests we've received followed by the number of rejected piece requests for various reasons. max_piece_requests mean we already had too many outstanding requests from this peer, so we rejected it. cancelled_piece_requests are ones where the other end explicitly asked for the piece to be rejected.
| name | type | 
|---|---|
| peer.error_incoming_peers | counter | 
| peer.error_outgoing_peers | counter | 
these counters break down the peer errors into whether they happen on incoming or outgoing peers.
| name | type | 
|---|---|
| peer.error_rc4_peers | counter | 
| peer.error_encrypted_peers | counter | 
these counters break down the peer errors into whether they happen on encrypted peers (just encrypted handshake) and rc4 peers (full stream encryption). These can indicate whether encrypted peers are more or less likely to fail
| name | type | 
|---|---|
| peer.error_tcp_peers | counter | 
| peer.error_utp_peers | counter | 
these counters break down the peer errors into whether they happen on uTP peers or TCP peers. these may indicate whether one protocol is more error prone
| name | type | 
|---|---|
| peer.connect_timeouts | counter | 
| peer.uninteresting_peers | counter | 
| peer.timeout_peers | counter | 
| peer.no_memory_peers | counter | 
| peer.too_many_peers | counter | 
| peer.transport_timeout_peers | counter | 
| peer.num_banned_peers | counter | 
| peer.banned_for_hash_failure | counter | 
| peer.connection_attempts | counter | 
| peer.connection_attempt_loops | counter | 
| peer.boost_connection_attempts | counter | 
| peer.missed_connection_attempts | counter | 
| peer.no_peer_connection_attempts | counter | 
| peer.incoming_connections | counter | 
these counters break down the reasons to disconnect peers.
| name | type | 
|---|---|
| peer.num_tcp_peers | gauge | 
| peer.num_socks5_peers | gauge | 
| peer.num_http_proxy_peers | gauge | 
| peer.num_utp_peers | gauge | 
| peer.num_i2p_peers | gauge | 
| peer.num_ssl_peers | gauge | 
| peer.num_ssl_socks5_peers | gauge | 
| peer.num_ssl_http_proxy_peers | gauge | 
| peer.num_ssl_utp_peers | gauge | 
| peer.num_peers_half_open | gauge | 
| peer.num_peers_connected | gauge | 
| peer.num_peers_up_interested | gauge | 
| peer.num_peers_down_interested | gauge | 
| peer.num_peers_up_unchoked_all | gauge | 
| peer.num_peers_up_unchoked_optimistic | gauge | 
| peer.num_peers_up_unchoked | gauge | 
| peer.num_peers_down_unchoked | gauge | 
| peer.num_peers_up_requests | gauge | 
| peer.num_peers_down_requests | gauge | 
| peer.num_peers_end_game | gauge | 
| peer.num_peers_up_disk | gauge | 
| peer.num_peers_down_disk | gauge | 
the number of peer connections for each kind of socket. num_peers_half_open counts half-open (connecting) peers, no other count includes those peers. num_peers_up_unchoked_all is the total number of unchoked peers, whereas num_peers_up_unchoked only are unchoked peers that count against the limit (i.e. excluding peers that are unchoked because the limit doesn't apply to them). num_peers_up_unchoked_optimistic is the number of optimistically unchoked peers.
| name | type | 
|---|---|
| net.on_read_counter | counter | 
| net.on_write_counter | counter | 
| net.on_tick_counter | counter | 
| net.on_lsd_counter | counter | 
| net.on_lsd_peer_counter | counter | 
| net.on_udp_counter | counter | 
| net.on_accept_counter | counter | 
| net.on_disk_queue_counter | counter | 
| net.on_disk_counter | counter | 
These counters count the number of times the network thread wakes up for each respective reason. If these counters are very large, it may indicate a performance issue, causing the network thread to wake up too ofte, wasting CPU. mitigate it by increasing buffers and limits for the specific trigger that wakes up the thread.
| name | type | 
|---|---|
| net.sent_payload_bytes | counter | 
| net.sent_bytes | counter | 
| net.sent_ip_overhead_bytes | counter | 
| net.sent_tracker_bytes | counter | 
| net.recv_payload_bytes | counter | 
| net.recv_bytes | counter | 
| net.recv_ip_overhead_bytes | counter | 
| net.recv_tracker_bytes | counter | 
total number of bytes sent and received by the session
| name | type | 
|---|---|
| net.limiter_up_queue | gauge | 
| net.limiter_down_queue | gauge | 
the number of sockets currently waiting for upload and download bandwidth from the rate limiter.
| name | type | 
|---|---|
| net.limiter_up_bytes | gauge | 
| net.limiter_down_bytes | gauge | 
the number of upload and download bytes waiting to be handed out from the rate limiter.
| name | type | 
|---|---|
| net.recv_failed_bytes | counter | 
the number of bytes downloaded that had to be discarded because they failed the hash check
| name | type | 
|---|---|
| net.recv_redundant_bytes | counter | 
the number of downloaded bytes that were discarded because they were downloaded multiple times (from different peers)
| name | type | 
|---|---|
| net.has_incoming_connections | gauge | 
is false by default and set to true when the first incoming connection is established this is used to know if the client is behind NAT or not.
| name | type | 
|---|---|
| ses.num_checking_torrents | gauge | 
| ses.num_stopped_torrents | gauge | 
| ses.num_upload_only_torrents | gauge | 
| ses.num_downloading_torrents | gauge | 
| ses.num_seeding_torrents | gauge | 
| ses.num_queued_seeding_torrents | gauge | 
| ses.num_queued_download_torrents | gauge | 
| ses.num_error_torrents | gauge | 
these gauges count the number of torrents in different states. Each torrent only belongs to one of these states. For torrents that could belong to multiple of these, the most prominent in picked. For instance, a torrent with an error counts as an error-torrent, regardless of its other state.
| name | type | 
|---|---|
| ses.non_filter_torrents | gauge | 
the number of torrents that don't have the IP filter applied to them.
| name | type | 
|---|---|
| ses.num_piece_passed | counter | 
| ses.num_piece_failed | counter | 
| ses.num_have_pieces | counter | 
| ses.num_total_pieces_added | counter | 
these count the number of times a piece has passed the hash check, the number of times a piece was successfully written to disk and the number of total possible pieces added by adding torrents. e.g. when adding a torrent with 1000 piece, num_total_pieces_added is incremented by 1000.
| name | type | 
|---|---|
| ses.num_unchoke_slots | gauge | 
the number of allowed unchoked peers
| name | type | 
|---|---|
| ses.num_outstanding_accept | gauge | 
the number of listen sockets that are currently accepting incoming connections
| name | type | 
|---|---|
| ses.num_incoming_choke | counter | 
| ses.num_incoming_unchoke | counter | 
| ses.num_incoming_interested | counter | 
| ses.num_incoming_not_interested | counter | 
| ses.num_incoming_have | counter | 
| ses.num_incoming_bitfield | counter | 
| ses.num_incoming_request | counter | 
| ses.num_incoming_piece | counter | 
| ses.num_incoming_cancel | counter | 
| ses.num_incoming_dht_port | counter | 
| ses.num_incoming_suggest | counter | 
| ses.num_incoming_have_all | counter | 
| ses.num_incoming_have_none | counter | 
| ses.num_incoming_reject | counter | 
| ses.num_incoming_allowed_fast | counter | 
| ses.num_incoming_ext_handshake | counter | 
| ses.num_incoming_pex | counter | 
| ses.num_incoming_metadata | counter | 
| ses.num_incoming_extended | counter | 
| ses.num_outgoing_choke | counter | 
| ses.num_outgoing_unchoke | counter | 
| ses.num_outgoing_interested | counter | 
| ses.num_outgoing_not_interested | counter | 
| ses.num_outgoing_have | counter | 
| ses.num_outgoing_bitfield | counter | 
| ses.num_outgoing_request | counter | 
| ses.num_outgoing_piece | counter | 
| ses.num_outgoing_cancel | counter | 
| ses.num_outgoing_dht_port | counter | 
| ses.num_outgoing_suggest | counter | 
| ses.num_outgoing_have_all | counter | 
| ses.num_outgoing_have_none | counter | 
| ses.num_outgoing_reject | counter | 
| ses.num_outgoing_allowed_fast | counter | 
| ses.num_outgoing_ext_handshake | counter | 
| ses.num_outgoing_pex | counter | 
| ses.num_outgoing_metadata | counter | 
| ses.num_outgoing_extended | counter | 
| ses.num_outgoing_hash_request | counter | 
| ses.num_outgoing_hashes | counter | 
| ses.num_outgoing_hash_reject | counter | 
bittorrent message counters. These counters are incremented every time a message of the corresponding type is received from or sent to a bittorrent peer.
| name | type | 
|---|---|
| ses.waste_piece_timed_out | counter | 
| ses.waste_piece_cancelled | counter | 
| ses.waste_piece_unknown | counter | 
| ses.waste_piece_seed | counter | 
| ses.waste_piece_end_game | counter | 
| ses.waste_piece_closing | counter | 
the number of wasted downloaded bytes by reason of the bytes being wasted.
| name | type | 
|---|---|
| picker.piece_picker_partial_loops | counter | 
| picker.piece_picker_suggest_loops | counter | 
| picker.piece_picker_sequential_loops | counter | 
| picker.piece_picker_reverse_rare_loops | counter | 
| picker.piece_picker_rare_loops | counter | 
| picker.piece_picker_rand_start_loops | counter | 
| picker.piece_picker_rand_loops | counter | 
| picker.piece_picker_busy_loops | counter | 
the number of pieces considered while picking pieces
| name | type | 
|---|---|
| picker.reject_piece_picks | counter | 
| picker.unchoke_piece_picks | counter | 
| picker.incoming_redundant_piece_picks | counter | 
| picker.incoming_piece_picks | counter | 
| picker.end_game_piece_picks | counter | 
| picker.snubbed_piece_picks | counter | 
| picker.interesting_piece_picks | counter | 
| picker.hash_fail_piece_picks | counter | 
This breaks down the piece picks into the event that triggered it
| name | type | 
|---|---|
| disk.request_latency | gauge | 
| disk.disk_blocks_in_use | gauge | 
the number of microseconds it takes from receiving a request from a peer until we're sending the response back on the socket.
| name | type | 
|---|---|
| disk.queued_disk_jobs | gauge | 
| disk.num_running_disk_jobs | gauge | 
| disk.num_read_jobs | gauge | 
| disk.num_write_jobs | gauge | 
| disk.num_jobs | gauge | 
| disk.blocked_disk_jobs | gauge | 
| disk.num_writing_threads | gauge | 
| disk.num_running_threads | gauge | 
queued_disk_jobs is the number of disk jobs currently queued, waiting to be executed by a disk thread.
| name | type | 
|---|---|
| disk.queued_write_bytes | gauge | 
the number of bytes we have sent to the disk I/O thread for writing. Every time we hear back from the disk I/O thread with a completed write job, this is updated to the number of bytes the disk I/O thread is actually waiting for to be written (as opposed to bytes just hanging out in the cache)
| name | type | 
|---|---|
| disk.num_blocks_written | counter | 
| disk.num_blocks_read | counter | 
the number of blocks written and read from disk in total. A block is 16 kiB. num_blocks_written and num_blocks_read
| name | type | 
|---|---|
| disk.num_blocks_hashed | counter | 
the total number of blocks run through SHA-1 hashing
| name | type | 
|---|---|
| disk.num_write_ops | counter | 
| disk.num_read_ops | counter | 
the number of disk I/O operation for reads and writes. One disk operation may transfer more then one block.
| name | type | 
|---|---|
| disk.num_read_back | counter | 
the number of blocks that had to be read back from disk in order to hash a piece (when verifying against the piece hash)
| name | type | 
|---|---|
| disk.disk_read_time | counter | 
| disk.disk_write_time | counter | 
| disk.disk_hash_time | counter | 
| disk.disk_job_time | counter | 
cumulative time spent in various disk jobs, as well as total for all disk jobs. Measured in microseconds
| name | type | 
|---|---|
| disk.num_fenced_read | gauge | 
| disk.num_fenced_write | gauge | 
| disk.num_fenced_hash | gauge | 
| disk.num_fenced_move_storage | gauge | 
| disk.num_fenced_release_files | gauge | 
| disk.num_fenced_delete_files | gauge | 
| disk.num_fenced_check_fastresume | gauge | 
| disk.num_fenced_save_resume_data | gauge | 
| disk.num_fenced_rename_file | gauge | 
| disk.num_fenced_stop_torrent | gauge | 
| disk.num_fenced_flush_piece | gauge | 
| disk.num_fenced_flush_hashed | gauge | 
| disk.num_fenced_flush_storage | gauge | 
| disk.num_fenced_file_priority | gauge | 
| disk.num_fenced_load_torrent | gauge | 
| disk.num_fenced_clear_piece | gauge | 
| disk.num_fenced_tick_storage | gauge | 
for each kind of disk job, a counter of how many jobs of that kind are currently blocked by a disk fence
| name | type | 
|---|---|
| dht.dht_nodes | gauge | 
The number of nodes in the DHT routing table
| name | type | 
|---|---|
| dht.dht_node_cache | gauge | 
The number of replacement nodes in the DHT routing table
| name | type | 
|---|---|
| dht.dht_torrents | gauge | 
the number of torrents currently tracked by our DHT node
| name | type | 
|---|---|
| dht.dht_peers | gauge | 
the number of peers currently tracked by our DHT node
| name | type | 
|---|---|
| dht.dht_immutable_data | gauge | 
the number of immutable data items tracked by our DHT node
| name | type | 
|---|---|
| dht.dht_mutable_data | gauge | 
the number of mutable data items tracked by our DHT node
| name | type | 
|---|---|
| dht.dht_allocated_observers | gauge | 
the number of RPC observers currently allocated
| name | type | 
|---|---|
| dht.dht_messages_in | counter | 
| dht.dht_messages_out | counter | 
the total number of DHT messages sent and received
| name | type | 
|---|---|
| dht.dht_messages_in_dropped | counter | 
the number of incoming DHT requests that were dropped. There are a few different reasons why incoming DHT packets may be dropped:
| name | type | 
|---|---|
| dht.dht_messages_out_dropped | counter | 
the number of outgoing messages that failed to be sent
| name | type | 
|---|---|
| dht.dht_bytes_in | counter | 
| dht.dht_bytes_out | counter | 
the total number of bytes sent and received by the DHT
| name | type | 
|---|---|
| dht.dht_ping_in | counter | 
| dht.dht_ping_out | counter | 
| dht.dht_find_node_in | counter | 
| dht.dht_find_node_out | counter | 
| dht.dht_get_peers_in | counter | 
| dht.dht_get_peers_out | counter | 
| dht.dht_announce_peer_in | counter | 
| dht.dht_announce_peer_out | counter | 
| dht.dht_get_in | counter | 
| dht.dht_get_out | counter | 
| dht.dht_put_in | counter | 
| dht.dht_put_out | counter | 
| dht.dht_sample_infohashes_in | counter | 
| dht.dht_sample_infohashes_out | counter | 
the number of DHT messages we've sent and received by kind.
| name | type | 
|---|---|
| dht.dht_invalid_announce | counter | 
| dht.dht_invalid_get_peers | counter | 
| dht.dht_invalid_find_node | counter | 
| dht.dht_invalid_put | counter | 
| dht.dht_invalid_get | counter | 
| dht.dht_invalid_sample_infohashes | counter | 
the number of failed incoming DHT requests by kind of request
| name | type | 
|---|---|
| utp.utp_packet_loss | counter | 
The number of times a lost packet has been interpreted as congestion, cutting the congestion window in half. Some lost packets are not interpreted as congestion, notably MTU-probes
| name | type | 
|---|---|
| utp.utp_timeout | counter | 
The number of timeouts experienced. This is when a connection doesn't hear back from the other end within a sliding average RTT + 2 average deviations from the mean (approximately). The actual time out is configurable and also depends on the state of the socket.
| name | type | 
|---|---|
| utp.utp_packets_in | counter | 
| utp.utp_packets_out | counter | 
The total number of packets sent and received
| name | type | 
|---|---|
| utp.utp_fast_retransmit | counter | 
The number of packets lost but re-sent by the fast-retransmit logic. This logic is triggered after 3 duplicate ACKs.
| name | type | 
|---|---|
| utp.utp_packet_resend | counter | 
The number of packets that were re-sent, for whatever reason
| name | type | 
|---|---|
| utp.utp_samples_above_target | counter | 
| utp.utp_samples_below_target | counter | 
The number of incoming packets where the delay samples were above and below the delay target, respectively. The delay target is configurable and is a parameter to the LEDBAT congestion control.
| name | type | 
|---|---|
| utp.utp_payload_pkts_in | counter | 
| utp.utp_payload_pkts_out | counter | 
The total number of packets carrying payload received and sent, respectively.
| name | type | 
|---|---|
| utp.utp_invalid_pkts_in | counter | 
The number of packets received that are not valid uTP packets (but were sufficiently similar to not be treated as DHT or UDP tracker packets).
| name | type | 
|---|---|
| utp.utp_redundant_pkts_in | counter | 
The number of duplicate payload packets received. This may happen if the outgoing ACK is lost.
| name | type | 
|---|---|
| utp.num_utp_idle | gauge | 
| utp.num_utp_syn_sent | gauge | 
| utp.num_utp_connected | gauge | 
| utp.num_utp_fin_sent | gauge | 
| utp.num_utp_close_wait | gauge | 
| utp.num_utp_deleted | gauge | 
the number of uTP sockets in each respective state
| name | type | 
|---|---|
| sock_bufs.socket_send_size3 | counter | 
| sock_bufs.socket_send_size4 | counter | 
| sock_bufs.socket_send_size5 | counter | 
| sock_bufs.socket_send_size6 | counter | 
| sock_bufs.socket_send_size7 | counter | 
| sock_bufs.socket_send_size8 | counter | 
| sock_bufs.socket_send_size9 | counter | 
| sock_bufs.socket_send_size10 | counter | 
| sock_bufs.socket_send_size11 | counter | 
| sock_bufs.socket_send_size12 | counter | 
| sock_bufs.socket_send_size13 | counter | 
| sock_bufs.socket_send_size14 | counter | 
| sock_bufs.socket_send_size15 | counter | 
| sock_bufs.socket_send_size16 | counter | 
| sock_bufs.socket_send_size17 | counter | 
| sock_bufs.socket_send_size18 | counter | 
| sock_bufs.socket_send_size19 | counter | 
| sock_bufs.socket_send_size20 | counter | 
| sock_bufs.socket_recv_size3 | counter | 
| sock_bufs.socket_recv_size4 | counter | 
| sock_bufs.socket_recv_size5 | counter | 
| sock_bufs.socket_recv_size6 | counter | 
| sock_bufs.socket_recv_size7 | counter | 
| sock_bufs.socket_recv_size8 | counter | 
| sock_bufs.socket_recv_size9 | counter | 
| sock_bufs.socket_recv_size10 | counter | 
| sock_bufs.socket_recv_size11 | counter | 
| sock_bufs.socket_recv_size12 | counter | 
| sock_bufs.socket_recv_size13 | counter | 
| sock_bufs.socket_recv_size14 | counter | 
| sock_bufs.socket_recv_size15 | counter | 
| sock_bufs.socket_recv_size16 | counter | 
| sock_bufs.socket_recv_size17 | counter | 
| sock_bufs.socket_recv_size18 | counter | 
| sock_bufs.socket_recv_size19 | counter | 
| sock_bufs.socket_recv_size20 | counter | 
the buffer sizes accepted by socket send and receive calls respectively. The larger the buffers are, the more efficient, because it require fewer system calls per byte. The size is 1 << n, where n is the number at the end of the counter name. i.e. 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096, 8192, 16384, 32768, 65536, 131072, 262144, 524288, 1048576 bytes
| name | type | 
|---|---|
| tracker.num_queued_tracker_announces | gauge | 
if the outstanding tracker announce limit is reached, tracker announces are queued, to be issued when an announce slot opens up. this measure the number of tracker announces currently in the queue
The libtorrent documentation use words that are bittorrent terms of art. This section defines some of these words. For an overview of what bittorrent is and how it works, see these slides. For an introduction to the bittorrent DHT, see this presentation.