Gazelle/classes/torrent.class.php

333 lines
9.4 KiB
PHP
Raw Normal View History

2011-03-28 14:21:28 +00:00
<?
/*******************************************************************************
|~~~~ Gazelle bencode parser ~~~~|
--------------------------------------------------------------------------------
Welcome to the Gazelle bencode parser. bencoding is the way of encoding data
that bittorrent uses in torrent files. When we read the torrent files, we get
one long string that must be parsed into a format we can easily edit - that's
where this file comes into play.
There are 4 data types in bencode:
* String
* Int
2013-02-22 08:00:24 +00:00
* List - array without keys
2011-03-28 14:21:28 +00:00
- like array('value', 'value 2', 'value 3', 'etc')
* Dictionary - array with string keys
- like array['key 1'] = 'value 1'; array['key 2'] = 'value 2';
Before you go any further, we recommend reading the sections on bencoding and
metainfo file structure here: http://wiki.theory.org/BitTorrentSpecification
//----- How we store the data -----//
* Strings
- Stored as php strings. Not difficult to remember.
* Integers
2013-02-22 08:00:24 +00:00
- Stored as php ints
2011-03-28 14:21:28 +00:00
- must be casted with (int)
* Lists
2013-02-22 08:00:24 +00:00
- Stored as a BENCODE_LIST object.
2011-03-28 14:21:28 +00:00
- The actual list is in BENCODE_LIST::$Val, as an array with incrementing integer indices
- The list in BENCODE_LIST::$Val is populated by the BENCODE_LIST::dec() function
* Dictionaries
2013-02-22 08:00:24 +00:00
- Stored as a BENCODE_DICT object.
2011-03-28 14:21:28 +00:00
- The actual list is in BENCODE_DICT::$Val, as an array with string indices
- The list in BENCODE_DICT::$Val is populated by the BENCODE_DICT::dec() function
//----- BENCODE_* Objects -----//
2013-02-22 08:00:24 +00:00
Lists and dictionaries are stored as objects. They each have the following
2011-03-28 14:21:28 +00:00
functions:
* decode(Type, $Key)
- Decodes ANY bencoded element, given the type and the key
- Gets the position and string from $this
* encode($Val)
- Encodes ANY non-bencoded element, given the value
* dec()
- Decodes either a dictionary or a list, depending on where it's called from
- Uses the decode() function quite a bit
* enc()
- Encodes either a dictionary or a list, depending on where it's called from
- Relies mostly on the encode() function
Finally, as all torrents are just large dictionaries, the TORRENT class extends
2013-02-22 08:00:24 +00:00
the BENCODE_DICT class.
2011-03-28 14:21:28 +00:00
*******************************************************************************/
2013-03-17 08:00:17 +00:00
class BENCODE2 {
2011-03-28 14:21:28 +00:00
var $Val; // Decoded array
var $Pos = 1; // Pointer that indicates our position in the string
var $Str = ''; // Torrent string
2013-02-22 08:00:24 +00:00
2013-04-20 08:01:01 +00:00
function __construct($Val, $IsParsed = false) {
if (!$IsParsed) {
2011-03-28 14:21:28 +00:00
$this->Str = $Val;
$this->dec();
} else {
$this->Val = $Val;
}
}
2013-02-22 08:00:24 +00:00
// Decode an element based on the type. The type is really just an indicator.
2013-04-20 08:01:01 +00:00
function decode($Type, $Key) {
if (is_number($Type)) { // Element is a string
2011-03-28 14:21:28 +00:00
// Get length of string
$StrLen = $Type;
2013-04-20 08:01:01 +00:00
while ($this->Str[$this->Pos + 1] != ':') {
2011-03-28 14:21:28 +00:00
$this->Pos++;
$StrLen.=$this->Str[$this->Pos];
}
2013-04-20 08:01:01 +00:00
$this->Val[$Key] = substr($this->Str, $this->Pos + 2, $StrLen);
2013-02-22 08:00:24 +00:00
2013-04-20 08:01:01 +00:00
$this->Pos += $StrLen;
$this->Pos += 2;
2013-02-22 08:00:24 +00:00
2013-04-20 08:01:01 +00:00
} elseif ($Type == 'i') { // Element is an int
2011-03-28 14:21:28 +00:00
$this->Pos++;
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
// Find end of integer (first occurance of 'e' after position)
2013-02-22 08:00:24 +00:00
$End = strpos($this->Str, 'e', $this->Pos);
2011-03-28 14:21:28 +00:00
// Get the integer, and - IMPORTANT - cast it as an int, so we know later that it's an int and not a string
2013-02-22 08:00:24 +00:00
$this->Val[$Key] = (int)substr($this->Str, $this->Pos, $End-$this->Pos);
2013-04-20 08:01:01 +00:00
$this->Pos = $End + 1;
2013-02-22 08:00:24 +00:00
2013-04-20 08:01:01 +00:00
} elseif ($Type == 'l') { // Element is a list
2011-03-28 14:21:28 +00:00
$this->Val[$Key] = new BENCODE_LIST(substr($this->Str, $this->Pos));
$this->Pos += $this->Val[$Key]->Pos;
2013-02-22 08:00:24 +00:00
2013-04-20 08:01:01 +00:00
} elseif ($Type == 'd') { // Element is a dictionary
2011-03-28 14:21:28 +00:00
$this->Val[$Key] = new BENCODE_DICT(substr($this->Str, $this->Pos));
$this->Pos += $this->Val[$Key]->Pos;
// Sort by key to respect spec
2013-02-18 08:00:22 +00:00
if (!empty($this->Val[$Key]->Val)) {
ksort($this->Val[$Key]->Val);
}
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
} else {
die('Invalid torrent file');
}
}
2013-02-22 08:00:24 +00:00
2013-04-20 08:01:01 +00:00
function encode($Val) {
if (is_int($Val)) { // Integer
2011-03-28 14:21:28 +00:00
return 'i'.$Val.'e';
2013-04-20 08:01:01 +00:00
} elseif (is_string($Val)) {
2011-03-28 14:21:28 +00:00
return strlen($Val).':'.$Val;
2013-04-20 08:01:01 +00:00
} elseif (is_object($Val)) {
2011-03-28 14:21:28 +00:00
return $Val->enc();
} else {
return 'fail';
}
}
}
2013-03-17 08:00:17 +00:00
class BENCODE_LIST extends BENCODE2 {
2013-04-20 08:01:01 +00:00
function enc() {
2013-02-18 08:00:22 +00:00
if (empty($this->Val)) {
2013-02-17 08:00:08 +00:00
return 'le';
}
2011-03-28 14:21:28 +00:00
$Str = 'l';
reset($this->Val);
foreach ($this->Val as $Value) {
$Str.=$this->encode($Value);
}
return $Str.'e';
}
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
// Decode a list
2013-04-20 08:01:01 +00:00
function dec() {
2011-03-28 14:21:28 +00:00
$Key = 0; // Array index
$Length = strlen($this->Str);
2013-04-20 08:01:01 +00:00
while ($this->Pos < $Length) {
2011-03-28 14:21:28 +00:00
$Type = $this->Str[$this->Pos];
// $Type now indicates what type of element we're dealing with
// It's either an integer (string), 'i' (an integer), 'l' (a list), 'd' (a dictionary), or 'e' (end of dictionary/list)
2013-02-22 08:00:24 +00:00
2013-04-20 08:01:01 +00:00
if ($Type == 'e') { // End of list
2011-03-28 14:21:28 +00:00
$this->Pos += 1;
unset($this->Str); // Since we're finished parsing the string, we don't need to store it anymore. Benchmarked - this makes the parser run way faster.
return;
}
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
// Decode the bencoded element.
// This function changes $this->Pos and $this->Val, so you don't have to.
$this->decode($Type, $Key);
2013-02-18 08:00:22 +00:00
++$Key;
2011-03-28 14:21:28 +00:00
}
return true;
}
}
2013-03-17 08:00:17 +00:00
class BENCODE_DICT extends BENCODE2 {
2013-04-20 08:01:01 +00:00
function enc() {
2013-02-18 08:00:22 +00:00
if (empty($this->Val)) {
return 'de';
}
2011-03-28 14:21:28 +00:00
$Str = 'd';
reset($this->Val);
foreach ($this->Val as $Key => $Value) {
$Str.=strlen($Key).':'.$Key.$this->encode($Value);
}
return $Str.'e';
}
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
// Decode a dictionary
2013-04-20 08:01:01 +00:00
function dec() {
2011-03-28 14:21:28 +00:00
$Length = strlen($this->Str);
2013-04-20 08:01:01 +00:00
while ($this->Pos<$Length) {
2013-02-22 08:00:24 +00:00
2013-04-20 08:01:01 +00:00
if ($this->Str[$this->Pos] == 'e') { // End of dictionary
2011-03-28 14:21:28 +00:00
$this->Pos += 1;
unset($this->Str); // Since we're finished parsing the string, we don't need to store it anymore. Benchmarked - this makes the parser run way faster.
return;
}
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
// Get the dictionary key
// Length of the key, in bytes
$KeyLen = $this->Str[$this->Pos];
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
// Allow for multi-digit lengths
2013-04-20 08:01:01 +00:00
while ($this->Str[$this->Pos + 1] != ':' && $this->Pos + 1 < $Length) {
2011-03-28 14:21:28 +00:00
$this->Pos++;
$KeyLen.=$this->Str[$this->Pos];
}
// $this->Pos is now on the last letter of the key length
// Adding 2 brings it past that character and the ':' to the beginning of the string
2013-04-20 08:01:01 +00:00
$this->Pos += 2;
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
// Get the name of the key
2013-02-22 08:00:24 +00:00
$Key = substr($this->Str, $this->Pos, $KeyLen);
2011-03-28 14:21:28 +00:00
// Move the position past the key to the beginning of the element
2013-04-20 08:01:01 +00:00
$this->Pos += $KeyLen;
2011-03-28 14:21:28 +00:00
$Type = $this->Str[$this->Pos];
// $Type now indicates what type of element we're dealing with
// It's either an integer (string), 'i' (an integer), 'l' (a list), 'd' (a dictionary), or 'e' (end of dictionary/list)
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
// Decode the bencoded element.
// This function changes $this->Pos and $this->Val, so you don't have to.
$this->decode($Type, $Key);
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
}
return true;
}
}
class TORRENT extends BENCODE_DICT {
function dump() {
// Convenience function used for testing and figuring out how we store the data
print_r($this->Val);
}
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
function dump_data() {
// Function which serializes $this->Val for storage
return base64_encode(serialize($this->Val));
}
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
/*
To use this, please remove the announce-list unset in make_private and be sure to still set_announce_url for backwards compatibility
function set_multi_announce() {
$Trackers = func_get_args();
$AnnounceList = new BENCODE_LIST(array(),true);
foreach ($Trackers as $Tracker) {
$SubList = new BENCODE_LIST(array($Tracker),true);
unset($SubList->Str);
$AnnounceList->Val[] = $SubList;
}
$this->Val['announce-list'] = $AnnounceList;
}
*/
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
function set_announce_url($Announce) {
$this->Val['announce'] = $Announce;
2012-11-21 08:00:08 +00:00
ksort($this->Val);
2011-03-28 14:21:28 +00:00
}
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
// Returns an array of:
2013-02-22 08:00:24 +00:00
// * the files in the torrent
2011-03-28 14:21:28 +00:00
// * the total size of files described therein
function file_list() {
$FileList = array();
2012-10-09 08:00:17 +00:00
if (!isset($this->Val['info']->Val['files'])) { // Single file mode
2011-03-28 14:21:28 +00:00
$TotalSize = $this->Val['info']->Val['length'];
2012-10-09 08:00:17 +00:00
$FileList[] = array($TotalSize, $this->get_name());
2011-03-28 14:21:28 +00:00
} else { // Multiple file mode
$FileNames = array();
2012-08-29 08:00:17 +00:00
$FileSizes = array();
2011-03-28 14:21:28 +00:00
$TotalSize = 0;
$Files = $this->Val['info']->Val['files']->Val;
2012-10-09 08:00:17 +00:00
if (isset($Files[0]->Val['path.utf-8'])) {
$PathKey = 'path.utf-8';
} else {
$PathKey = 'path';
}
foreach ($Files as $File) {
2011-03-28 14:21:28 +00:00
$FileSize = $File->Val['length'];
2012-10-09 08:00:17 +00:00
$TotalSize += $FileSize;
2013-02-22 08:00:24 +00:00
2012-10-09 08:00:17 +00:00
$FileName = ltrim(implode('/',$File->Val[$PathKey]->Val), '/');
2012-08-29 08:00:17 +00:00
$FileSizes[] = $FileSize;
2011-03-28 14:21:28 +00:00
$FileNames[] = $FileName;
}
2012-08-29 08:00:17 +00:00
natcasesort($FileNames);
2012-10-09 08:00:17 +00:00
foreach ($FileNames as $Index => $FileName) {
2012-08-29 08:00:17 +00:00
$FileList[] = array($FileSizes[$Index], $FileName);
}
2011-03-28 14:21:28 +00:00
}
return array($TotalSize, $FileList);
}
2012-10-09 08:00:17 +00:00
function get_name() {
if (isset($this->Val['info']->Val['name.utf-8'])) {
return $this->Val['info']->Val['name.utf-8'];
} else {
return $this->Val['info']->Val['name'];
}
}
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
function make_private() {
//----- The following properties do not affect the infohash:
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
// anounce-list is an unofficial extension to the protocol
// that allows for multiple trackers per torrent
unset($this->Val['announce-list']);
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
// Bitcomet & Azureus cache peers in here
unset($this->Val['nodes']);
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
// Azureus stores the dht_backup_enable flag here
unset($this->Val['azureus_properties']);
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
// Remove web-seeds
unset($this->Val['url-list']);
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
// Remove libtorrent resume info
unset($this->Val['libtorrent_resume']);
2013-02-22 08:00:24 +00:00
2011-03-28 14:21:28 +00:00
//----- End properties that do not affect the infohash
if ($this->Val['info']->Val['private']) {
return true; // Torrent is private
} else {
// Torrent is not private!
2011-11-08 08:00:26 +00:00
// add private tracker flag and sort info dictionary
2011-03-28 14:21:28 +00:00
$this->Val['info']->Val['private'] = 1;
2011-11-08 08:00:26 +00:00
ksort($this->Val['info']->Val);
2011-03-28 14:21:28 +00:00
return false;
}
}
}
?>