| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006 | #include "bittorrent_helper.h"#include <cstring>#include <iostream>#include <cppunit/extensions/HelperMacros.h>#include "DownloadContext.h"#include "util.h"#include "RecoverableException.h"#include "AnnounceTier.h"#include "FixedNumberRandomizer.h"#include "FileEntry.h"#include "array_fun.h"#include "a2netcompat.h"#include "bencode2.h"#include "TestUtil.h"#include "base32.h"#include "Option.h"#include "prefs.h"namespace aria2 {namespace bittorrent {class BittorrentHelperTest:public CppUnit::TestFixture {  CPPUNIT_TEST_SUITE(BittorrentHelperTest);  CPPUNIT_TEST(testGetInfoHash);  CPPUNIT_TEST(testGetPieceHash);  CPPUNIT_TEST(testGetFileEntries);  CPPUNIT_TEST(testGetTotalLength);  CPPUNIT_TEST(testGetFileEntriesSingle);  CPPUNIT_TEST(testGetTotalLengthSingle);  CPPUNIT_TEST(testGetFileModeMulti);  CPPUNIT_TEST(testGetFileModeSingle);  CPPUNIT_TEST(testGetNameMulti);  CPPUNIT_TEST(testGetNameSingle);  CPPUNIT_TEST(testOverrideName);  CPPUNIT_TEST(testGetAnnounceTier);  CPPUNIT_TEST(testGetAnnounceTierAnnounceList);  CPPUNIT_TEST(testGetPieceLength);  CPPUNIT_TEST(testGetInfoHashAsString);  CPPUNIT_TEST(testGetPeerId);  CPPUNIT_TEST(testComputeFastSet);  CPPUNIT_TEST(testGetFileEntries_multiFileUrlList);  CPPUNIT_TEST(testGetFileEntries_singleFileUrlList);  CPPUNIT_TEST(testGetFileEntries_singleFileUrlListEndsWithSlash);  CPPUNIT_TEST(testLoadFromMemory);  CPPUNIT_TEST(testLoadFromMemory_somethingMissing);  CPPUNIT_TEST(testLoadFromMemory_overrideName);  CPPUNIT_TEST(testLoadFromMemory_multiFileDirTraversal);  CPPUNIT_TEST(testLoadFromMemory_singleFileDirTraversal);  CPPUNIT_TEST(testLoadFromMemory_multiFileNonUtf8Path);  CPPUNIT_TEST(testLoadFromMemory_singleFileNonUtf8Path);  CPPUNIT_TEST(testGetNodes);  CPPUNIT_TEST(testGetBasePath);  CPPUNIT_TEST(testSetFileFilter_single);  CPPUNIT_TEST(testSetFileFilter_multi);  CPPUNIT_TEST(testUTF8Torrent);  CPPUNIT_TEST(testEtc);  CPPUNIT_TEST(testCheckBitfield);  CPPUNIT_TEST(testMetadata);  CPPUNIT_TEST(testParseMagnet);  CPPUNIT_TEST(testParseMagnet_base32);  CPPUNIT_TEST(testMetadata2Torrent);  CPPUNIT_TEST(testTorrent2Magnet);  CPPUNIT_TEST(testExtractPeerFromString);  CPPUNIT_TEST(testExtractPeerFromList);  CPPUNIT_TEST(testExtract2PeersFromList);  CPPUNIT_TEST(testPackcompact);  CPPUNIT_TEST(testUnpackcompact);  CPPUNIT_TEST(testRemoveAnnounceUri);  CPPUNIT_TEST(testAddAnnounceUri);  CPPUNIT_TEST(testAdjustAnnounceUri);  CPPUNIT_TEST_SUITE_END();public:  SharedHandle<Option> option_;  void setUp()  {    option_.reset(new Option());    option_->put(PREF_DIR, ".");  }  void testGetInfoHash();  void testGetPieceHash();  void testGetFileEntries();  void testGetTotalLength();  void testGetFileEntriesSingle();  void testGetTotalLengthSingle();  void testGetFileModeMulti();  void testGetFileModeSingle();  void testGetNameMulti();  void testGetNameSingle();  void testOverrideName();  void testGetAnnounceTier();  void testGetAnnounceTierAnnounceList();  void testGetPieceLength();  void testGetInfoHashAsString();  void testGetPeerId();  void testComputeFastSet();  void testGetFileEntries_multiFileUrlList();  void testGetFileEntries_singleFileUrlList();  void testGetFileEntries_singleFileUrlListEndsWithSlash();  void testLoadFromMemory();  void testLoadFromMemory_somethingMissing();  void testLoadFromMemory_overrideName();  void testLoadFromMemory_multiFileDirTraversal();  void testLoadFromMemory_singleFileDirTraversal();  void testLoadFromMemory_multiFileNonUtf8Path();  void testLoadFromMemory_singleFileNonUtf8Path();  void testGetNodes();  void testGetBasePath();  void testSetFileFilter_single();  void testSetFileFilter_multi();  void testUTF8Torrent();  void testEtc();  void testCheckBitfield();  void testMetadata();  void testParseMagnet();  void testParseMagnet_base32();  void testMetadata2Torrent();  void testTorrent2Magnet();  void testExtractPeerFromString();  void testExtractPeerFromList();  void testExtract2PeersFromList();  void testPackcompact();  void testUnpackcompact();  void testRemoveAnnounceUri();  void testAddAnnounceUri();  void testAdjustAnnounceUri();};CPPUNIT_TEST_SUITE_REGISTRATION(BittorrentHelperTest);void BittorrentHelperTest::testGetInfoHash() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/test.torrent", dctx, option_);  std::string correctHash = "248d0a1cd08284299de78d5c1ed359bb46717d8c";  CPPUNIT_ASSERT_EQUAL(correctHash, bittorrent::getInfoHashString(dctx));}void BittorrentHelperTest::testGetPieceHash() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/test.torrent", dctx, option_);  CPPUNIT_ASSERT_EQUAL(std::string("AAAAAAAAAAAAAAAAAAAA"),                       dctx->getPieceHash(0));  CPPUNIT_ASSERT_EQUAL(std::string("BBBBBBBBBBBBBBBBBBBB"),                       dctx->getPieceHash(1));  CPPUNIT_ASSERT_EQUAL(std::string("CCCCCCCCCCCCCCCCCCCC"),                       dctx->getPieceHash(2));  CPPUNIT_ASSERT_EQUAL(std::string(""),                       dctx->getPieceHash(3));  CPPUNIT_ASSERT_EQUAL(std::string("sha-1"), dctx->getPieceHashType());}void BittorrentHelperTest::testGetFileEntries() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  option_->put(PREF_MAX_CONNECTION_PER_SERVER, "10");  load(A2_TEST_DIR"/test.torrent", dctx, option_);  // This is multi-file torrent.  std::vector<SharedHandle<FileEntry> > fileEntries = dctx->getFileEntries();  // There are 2 file entries.  CPPUNIT_ASSERT_EQUAL((size_t)2, fileEntries.size());  std::vector<SharedHandle<FileEntry> >::iterator itr = fileEntries.begin();  SharedHandle<FileEntry> fileEntry1 = *itr;  CPPUNIT_ASSERT_EQUAL(std::string("./aria2-test/aria2/src/aria2c"),                       fileEntry1->getPath());  CPPUNIT_ASSERT_EQUAL(std::string("aria2-test/aria2/src/aria2c"),                       fileEntry1->getOriginalName());  CPPUNIT_ASSERT_EQUAL(10, fileEntry1->getMaxConnectionPerServer());  itr++;  SharedHandle<FileEntry> fileEntry2 = *itr;  CPPUNIT_ASSERT_EQUAL(std::string("./aria2-test/aria2-0.2.2.tar.bz2"),                       fileEntry2->getPath());  CPPUNIT_ASSERT_EQUAL(10, fileEntry2->getMaxConnectionPerServer());}void BittorrentHelperTest::testGetFileEntriesSingle() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  option_->put(PREF_MAX_CONNECTION_PER_SERVER, "10");  load(A2_TEST_DIR"/single.torrent", dctx, option_);  // This is multi-file torrent.  std::vector<SharedHandle<FileEntry> > fileEntries = dctx->getFileEntries();  // There is 1 file entry.  CPPUNIT_ASSERT_EQUAL((size_t)1, fileEntries.size());  std::vector<SharedHandle<FileEntry> >::iterator itr = fileEntries.begin();  SharedHandle<FileEntry> fileEntry1 = *itr;  CPPUNIT_ASSERT_EQUAL(std::string("./aria2-0.8.2.tar.bz2"),                       fileEntry1->getPath());  CPPUNIT_ASSERT_EQUAL(std::string("aria2-0.8.2.tar.bz2"),                       fileEntry1->getOriginalName());  CPPUNIT_ASSERT_EQUAL(10, fileEntry1->getMaxConnectionPerServer());}void BittorrentHelperTest::testGetTotalLength() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/test.torrent", dctx, option_);  CPPUNIT_ASSERT_EQUAL((int64_t)384LL, dctx->getTotalLength());}void BittorrentHelperTest::testGetTotalLengthSingle() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/single.torrent", dctx, option_);  CPPUNIT_ASSERT_EQUAL((int64_t)384LL, dctx->getTotalLength());}void BittorrentHelperTest::testGetFileModeMulti() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/test.torrent", dctx, option_);  CPPUNIT_ASSERT_EQUAL(MULTI, getTorrentAttrs(dctx)->mode);}void BittorrentHelperTest::testGetFileModeSingle() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/single.torrent", dctx, option_);  CPPUNIT_ASSERT_EQUAL(SINGLE, getTorrentAttrs(dctx)->mode);}void BittorrentHelperTest::testGetNameMulti() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/test.torrent", dctx, option_);  CPPUNIT_ASSERT_EQUAL(std::string("aria2-test"), getTorrentAttrs(dctx)->name);}void BittorrentHelperTest::testGetNameSingle() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/single.torrent", dctx, option_);  CPPUNIT_ASSERT_EQUAL(std::string("./aria2-0.8.2.tar.bz2"),                       dctx->getBasePath());  CPPUNIT_ASSERT_EQUAL(std::string("aria2-0.8.2.tar.bz2"),                       getTorrentAttrs(dctx)->name);}void BittorrentHelperTest::testOverrideName(){  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/test.torrent", dctx, option_, "aria2-override.name");  CPPUNIT_ASSERT_EQUAL(std::string("./aria2-override.name"),                       dctx->getBasePath());  CPPUNIT_ASSERT_EQUAL(std::string("aria2-override.name"),                       getTorrentAttrs(dctx)->name);}void BittorrentHelperTest::testGetAnnounceTier() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/single.torrent", dctx, option_);  SharedHandle<TorrentAttribute> attrs = getTorrentAttrs(dctx);  // There is 1 tier.  CPPUNIT_ASSERT_EQUAL((size_t)1, attrs->announceList.size());  CPPUNIT_ASSERT_EQUAL((size_t)1, attrs->announceList[0].size());  CPPUNIT_ASSERT_EQUAL(std::string("http://aria.rednoah.com/announce.php"),                       attrs->announceList[0][0]);}void BittorrentHelperTest::testGetAnnounceTierAnnounceList() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/test.torrent", dctx, option_);  SharedHandle<TorrentAttribute> attrs = getTorrentAttrs(dctx);  // There are 3 tiers.  CPPUNIT_ASSERT_EQUAL((size_t)3, attrs->announceList.size());  CPPUNIT_ASSERT_EQUAL((size_t)1, attrs->announceList[0].size());  CPPUNIT_ASSERT_EQUAL(std::string("http://tracker1"),                       attrs->announceList[0][0]);  CPPUNIT_ASSERT_EQUAL((size_t)1, attrs->announceList[1].size());  CPPUNIT_ASSERT_EQUAL(std::string("http://tracker2"),                       attrs->announceList[1][0]);  CPPUNIT_ASSERT_EQUAL((size_t)1, attrs->announceList[2].size());  CPPUNIT_ASSERT_EQUAL(std::string("http://tracker3"),                       attrs->announceList[2][0]);}void BittorrentHelperTest::testGetPieceLength() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/test.torrent", dctx, option_);  CPPUNIT_ASSERT_EQUAL(128, dctx->getPieceLength());}void BittorrentHelperTest::testGetInfoHashAsString() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/test.torrent", dctx, option_);  CPPUNIT_ASSERT_EQUAL(std::string("248d0a1cd08284299de78d5c1ed359bb46717d8c"),                       getInfoHashString(dctx));}void BittorrentHelperTest::testGetPeerId() {  std::string peerId = generatePeerId("aria2-");  CPPUNIT_ASSERT(peerId.find("aria2-") == 0);  CPPUNIT_ASSERT_EQUAL((size_t)20, peerId.size());}void BittorrentHelperTest::testComputeFastSet(){  std::string ipaddr = "192.168.0.1";  unsigned char infoHash[20];  memset(infoHash, 0, sizeof(infoHash));  infoHash[0] = 0xff;  int fastSetSize = 10;  size_t numPieces = 1000;  {    std::vector<size_t> fastSet;    computeFastSet(fastSet, ipaddr, numPieces, infoHash, fastSetSize);    size_t ans[] = { 686, 459, 278, 200, 404, 834, 64, 203, 760, 950 };    std::vector<size_t> ansSet(vbegin(ans), vend(ans));    CPPUNIT_ASSERT(std::equal(fastSet.begin(), fastSet.end(), ansSet.begin()));  }  ipaddr = "10.0.0.1";  {    std::vector<size_t> fastSet;    computeFastSet(fastSet, ipaddr, numPieces, infoHash, fastSetSize);    size_t ans[] = { 568, 188, 466, 452, 550, 662, 109, 226, 398, 11 };    std::vector<size_t> ansSet(vbegin(ans), vend(ans));    CPPUNIT_ASSERT(std::equal(fastSet.begin(), fastSet.end(), ansSet.begin()));  }  // See when pieces < fastSetSize  numPieces = 9;  {    std::vector<size_t> fastSet;    computeFastSet(fastSet, ipaddr, numPieces, infoHash, fastSetSize);    size_t ans[] = { 8, 6, 7, 5, 1, 4, 0, 2, 3 };    std::vector<size_t> ansSet(vbegin(ans), vend(ans));    CPPUNIT_ASSERT(std::equal(fastSet.begin(), fastSet.end(), ansSet.begin()));  }}void BittorrentHelperTest::testGetFileEntries_multiFileUrlList() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/url-list-multiFile.torrent", dctx, option_);  // This is multi-file torrent.  const std::vector<SharedHandle<FileEntry> >& fileEntries =    dctx->getFileEntries();  // There are 2 file entries.  CPPUNIT_ASSERT_EQUAL((size_t)2, fileEntries.size());  std::vector<SharedHandle<FileEntry> >::const_iterator itr =    fileEntries.begin();  const SharedHandle<FileEntry>& fileEntry1 = *itr;  CPPUNIT_ASSERT_EQUAL(std::string("./aria2-test@/aria2@/src@/aria2c@"),                       fileEntry1->getPath());  const std::deque<std::string>& uris1 = fileEntry1->getRemainingUris();  CPPUNIT_ASSERT_EQUAL((size_t)2, uris1.size());  CPPUNIT_ASSERT_EQUAL(std::string("http://localhost/dist/aria2-test%40/aria2%40/src%40/aria2c%40"),                       uris1[0]);  CPPUNIT_ASSERT_EQUAL(std::string("http://mirror/dist/aria2-test%40/aria2%40/src%40/aria2c%40"),                       uris1[1]);  ++itr;  const SharedHandle<FileEntry>& fileEntry2 = *itr;  CPPUNIT_ASSERT_EQUAL(std::string("./aria2-test@/aria2-0.2.2.tar.bz2"),                       fileEntry2->getPath());  const std::deque<std::string>& uris2 = fileEntry2->getRemainingUris();  CPPUNIT_ASSERT_EQUAL((size_t)2, uris2.size());  CPPUNIT_ASSERT_EQUAL(std::string("http://localhost/dist/aria2-test%40/aria2-0.2.2.tar.bz2"),                       uris2[0]);  CPPUNIT_ASSERT_EQUAL(std::string("http://mirror/dist/aria2-test%40/aria2-0.2.2.tar.bz2"),                       uris2[1]);}void BittorrentHelperTest::testGetFileEntries_singleFileUrlList() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/url-list-singleFile.torrent", dctx, option_);  // This is single-file torrent.  const std::vector<SharedHandle<FileEntry> >& fileEntries =    dctx->getFileEntries();  // There are 1 file entries.  CPPUNIT_ASSERT_EQUAL((size_t)1, fileEntries.size());  const SharedHandle<FileEntry>& fileEntry1 = fileEntries.front();  CPPUNIT_ASSERT_EQUAL(std::string("./aria2.tar.bz2"),                       fileEntry1->getPath());  const std::deque<std::string>& uris1 = fileEntry1->getRemainingUris();  CPPUNIT_ASSERT_EQUAL((size_t)1, uris1.size());  CPPUNIT_ASSERT_EQUAL(std::string("http://localhost/dist/aria2.tar.bz2"),                       uris1[0]);}void BittorrentHelperTest::testGetFileEntries_singleFileUrlListEndsWithSlash() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/url-list-singleFileEndsWithSlash.torrent", dctx, option_);  // This is single-file torrent.  const std::vector<SharedHandle<FileEntry> >& fileEntries =    dctx->getFileEntries();  // There are 1 file entries.  CPPUNIT_ASSERT_EQUAL((size_t)1, fileEntries.size());  const SharedHandle<FileEntry>& fileEntry1 = fileEntries.front();  CPPUNIT_ASSERT_EQUAL(std::string("./aria2@.tar.bz2"),                       fileEntry1->getPath());  const std::deque<std::string>& uris1 = fileEntry1->getRemainingUris();  CPPUNIT_ASSERT_EQUAL((size_t)1, uris1.size());  CPPUNIT_ASSERT_EQUAL(std::string("http://localhost/dist/aria2%40.tar.bz2"),                       uris1[0]);}void BittorrentHelperTest::testLoadFromMemory_multiFileNonUtf8Path(){  SharedHandle<List> path = List::g();  path->append("path");  path->append(fromHex("90a28a")+"E");  SharedHandle<Dict> file = Dict::g();  file->put("length", Integer::g(1024));  file->put("path", path);  SharedHandle<List> files = List::g();  files->append(file);  SharedHandle<Dict> info = Dict::g();  info->put("files", files);  info->put("piece length", Integer::g(1024));  info->put("pieces", "01234567890123456789");  info->put("name", fromHex("1b")+"$B%O%m!<"+fromHex("1b")+"(B");  Dict dict;  dict.put("info", info);  SharedHandle<DownloadContext> dctx(new DownloadContext());  loadFromMemory(bencode2::encode(&dict), dctx, option_, "default");  const SharedHandle<FileEntry>& fe = dctx->getFirstFileEntry();  CPPUNIT_ASSERT_EQUAL    (std::string("./%1B%24B%25O%25m%21%3C%1B%28B/path/%90%A2%8AE"),     fe->getPath());  CPPUNIT_ASSERT_EQUAL    (std::string("./%1B%24B%25O%25m%21%3C%1B%28B"), dctx->getBasePath());}void BittorrentHelperTest::testLoadFromMemory_singleFileNonUtf8Path(){  SharedHandle<Dict> info = Dict::g();  info->put("piece length", Integer::g(1024));  info->put("pieces", "01234567890123456789");  info->put("name", fromHex("90a28a")+"E");  info->put("length", Integer::g(1024));  Dict dict;  dict.put("info", info);  SharedHandle<DownloadContext> dctx(new DownloadContext());  loadFromMemory(bencode2::encode(&dict), dctx, option_, "default");  const SharedHandle<FileEntry>& fe = dctx->getFirstFileEntry();  CPPUNIT_ASSERT_EQUAL(std::string("./%90%A2%8AE"), fe->getPath());}void BittorrentHelperTest::testLoadFromMemory(){  std::string memory = "d8:announce36:http://aria.rednoah.com/announce.php13:announce-listll16:http://tracker1 el15:http://tracker2el15:http://tracker3ee7:comment17:REDNOAH.COM RULES13:creation datei1123456789e4:infod5:filesld6:lengthi284e4:pathl5:aria23:src6:aria2ceed6:lengthi100e4:pathl19:aria2-0.2.2.tar.bz2eee4:name10:aria2-test12:piece lengthi128e6:pieces60:AAAAAAAAAAAAAAAAAAAABBBBBBBBBBBBBBBBBBBBCCCCCCCCCCCCCCCCCCCCee";  SharedHandle<DownloadContext> dctx(new DownloadContext());  loadFromMemory(memory, dctx, option_, "default");  std::string correctHash = "248d0a1cd08284299de78d5c1ed359bb46717d8c";  CPPUNIT_ASSERT_EQUAL(correctHash, getInfoHashString(dctx));}void BittorrentHelperTest::testLoadFromMemory_somethingMissing(){  // pieces missing  try {    std::string memory = "d8:announce36:http://aria.rednoah.com/announce.php4:infod4:name13:aria2.tar.bz26:lengthi262144eee";    SharedHandle<DownloadContext> dctx(new DownloadContext());    loadFromMemory(memory, dctx, option_, "default");    CPPUNIT_FAIL("exception must be thrown.");  } catch(Exception& e) {    // OK  }}void BittorrentHelperTest::testLoadFromMemory_overrideName(){  std::string memory = "d8:announce36:http://aria.rednoah.com/announce.php13:announce-listll16:http://tracker1 el15:http://tracker2el15:http://tracker3ee7:comment17:REDNOAH.COM RULES13:creation datei1123456789e4:infod5:filesld6:lengthi284e4:pathl5:aria23:src6:aria2ceed6:lengthi100e4:pathl19:aria2-0.2.2.tar.bz2eee4:name10:aria2-test12:piece lengthi128e6:pieces60:AAAAAAAAAAAAAAAAAAAABBBBBBBBBBBBBBBBBBBBCCCCCCCCCCCCCCCCCCCCee";  SharedHandle<DownloadContext> dctx(new DownloadContext());  loadFromMemory(memory, dctx, option_, "default", "aria2-override.name");  CPPUNIT_ASSERT_EQUAL(std::string("aria2-override.name"),                       getTorrentAttrs(dctx)->name);}void BittorrentHelperTest::testLoadFromMemory_multiFileDirTraversal(){  std::string memory =    "d8:announce27:http://example.com/announce4:infod5:filesld6:lengthi262144e4:pathl7:../dir14:dir28:file.imgeee4:name14:../name1/name212:piece lengthi262144e6:pieces20:00000000000000000000ee";  SharedHandle<DownloadContext> dctx(new DownloadContext());  try {    loadFromMemory(memory, dctx, option_, "default");    CPPUNIT_FAIL("Exception must be thrown.");  } catch(RecoverableException& e) {    // success  }}void BittorrentHelperTest::testLoadFromMemory_singleFileDirTraversal(){  std::string memory =    "d8:announce27:http://example.com/announce4:infod4:name14:../name1/name26:lengthi262144e12:piece lengthi262144e6:pieces20:00000000000000000000ee";  SharedHandle<DownloadContext> dctx(new DownloadContext());  try {    loadFromMemory(memory, dctx, option_, "default");  } catch(RecoverableException& e) {    // success  }}void BittorrentHelperTest::testGetNodes(){  {    std::string memory =      "d5:nodesl"      "l11:192.168.0.1i6881ee"      "l11:192.168.0.2i6882ee"      "e4:infod4:name13:aria2.tar.bz26:lengthi262144e"      "12:piece lengthi262144e"      "6:pieces20:AAAAAAAAAAAAAAAAAAAA"      "ee";    SharedHandle<DownloadContext> dctx(new DownloadContext());    loadFromMemory(memory, dctx, option_, "default");    SharedHandle<TorrentAttribute> attrs = getTorrentAttrs(dctx);    CPPUNIT_ASSERT_EQUAL((size_t)2, attrs->nodes.size());    CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.1"), attrs->nodes[0].first);    CPPUNIT_ASSERT_EQUAL((uint16_t)6881, attrs->nodes[0].second);    CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.2"), attrs->nodes[1].first);    CPPUNIT_ASSERT_EQUAL((uint16_t)6882, attrs->nodes[1].second);  }  {    // empty hostname    std::string memory =      "d5:nodesl"      "l1: i6881ee"      "l11:192.168.0.2i6882ee"      "e4:infod4:name13:aria2.tar.bz26:lengthi262144e"      "12:piece lengthi262144e"      "6:pieces20:AAAAAAAAAAAAAAAAAAAA"      "ee";    SharedHandle<DownloadContext> dctx(new DownloadContext());    loadFromMemory(memory, dctx, option_, "default");    SharedHandle<TorrentAttribute> attrs = getTorrentAttrs(dctx);    CPPUNIT_ASSERT_EQUAL((size_t)1, attrs->nodes.size());    CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.2"), attrs->nodes[0].first);    CPPUNIT_ASSERT_EQUAL((uint16_t)6882, attrs->nodes[0].second);  }  {    // bad port    std::string memory =      "d5:nodesl"      "l11:192.168.0.11:xe"      "l11:192.168.0.2i6882ee"      "e4:infod4:name13:aria2.tar.bz26:lengthi262144e"      "12:piece lengthi262144e"      "6:pieces20:AAAAAAAAAAAAAAAAAAAA"      "ee";    SharedHandle<DownloadContext> dctx(new DownloadContext());    loadFromMemory(memory, dctx, option_, "default");    SharedHandle<TorrentAttribute> attrs = getTorrentAttrs(dctx);    CPPUNIT_ASSERT_EQUAL((size_t)1, attrs->nodes.size());    CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.2"), attrs->nodes[0].first);    CPPUNIT_ASSERT_EQUAL((uint16_t)6882, attrs->nodes[0].second);  }  {    // port missing    std::string memory =      "d5:nodesl"      "l11:192.168.0.1e"      "l11:192.168.0.2i6882ee"      "e4:infod4:name13:aria2.tar.bz26:lengthi262144e"      "12:piece lengthi262144e"      "6:pieces20:AAAAAAAAAAAAAAAAAAAA"      "ee";    SharedHandle<DownloadContext> dctx(new DownloadContext());    loadFromMemory(memory, dctx, option_, "default");    SharedHandle<TorrentAttribute> attrs = getTorrentAttrs(dctx);    CPPUNIT_ASSERT_EQUAL((size_t)1, attrs->nodes.size());    CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.2"), attrs->nodes[0].first);    CPPUNIT_ASSERT_EQUAL((uint16_t)6882, attrs->nodes[0].second);  }  {    // nodes is not a list    std::string memory =      "d5:nodes"      "l11:192.168.0.1e"      "4:infod4:name13:aria2.tar.bz26:lengthi262144e"      "12:piece lengthi262144e"      "6:pieces20:AAAAAAAAAAAAAAAAAAAA"      "ee";    SharedHandle<DownloadContext> dctx(new DownloadContext());    loadFromMemory(memory, dctx, option_, "default");    SharedHandle<TorrentAttribute> attrs = getTorrentAttrs(dctx);    CPPUNIT_ASSERT_EQUAL((size_t)0, attrs->nodes.size());  }  {    // the element of node is not Data    std::string memory =      "d5:nodesl"      "ll11:192.168.0.1i6881eee"      "l11:192.168.0.2i6882ee"      "e4:infod4:name13:aria2.tar.bz26:lengthi262144e"      "12:piece lengthi262144e"      "6:pieces20:AAAAAAAAAAAAAAAAAAAA"      "ee";    SharedHandle<DownloadContext> dctx(new DownloadContext());    loadFromMemory(memory, dctx, option_, "default");    SharedHandle<TorrentAttribute> attrs = getTorrentAttrs(dctx);    CPPUNIT_ASSERT_EQUAL((size_t)1, attrs->nodes.size());    CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.2"), attrs->nodes[0].first);    CPPUNIT_ASSERT_EQUAL((uint16_t)6882, attrs->nodes[0].second);  }}void BittorrentHelperTest::testGetBasePath(){  SharedHandle<DownloadContext> singleCtx(new DownloadContext());  load(A2_TEST_DIR"/single.torrent", singleCtx, option_);  singleCtx->setFilePathWithIndex(1, "new-path");  CPPUNIT_ASSERT_EQUAL(std::string("new-path"), singleCtx->getBasePath());  option_->put(PREF_DIR, "downloads");  SharedHandle<DownloadContext> multiCtx(new DownloadContext());  load(A2_TEST_DIR"/test.torrent", multiCtx, option_);  CPPUNIT_ASSERT_EQUAL(std::string("downloads/aria2-test"),                       multiCtx->getBasePath());}void BittorrentHelperTest::testSetFileFilter_single(){  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/single.torrent", dctx, option_);  CPPUNIT_ASSERT(dctx->getFirstFileEntry()->isRequested());  SegList<int> sgl;  dctx->setFileFilter(sgl);  CPPUNIT_ASSERT(dctx->getFirstFileEntry()->isRequested());  sgl.clear();  sgl.add(1, 2);  dctx->setFileFilter(sgl);  CPPUNIT_ASSERT(dctx->getFirstFileEntry()->isRequested());  // For single file torrent, file is always selected whatever range  // is passed.  sgl.clear();  sgl.add(2, 3);  dctx->setFileFilter(sgl);  CPPUNIT_ASSERT(dctx->getFirstFileEntry()->isRequested());}void BittorrentHelperTest::testSetFileFilter_multi(){  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/test.torrent", dctx, option_);  CPPUNIT_ASSERT(dctx->getFileEntries()[0]->isRequested());  CPPUNIT_ASSERT(dctx->getFileEntries()[1]->isRequested());  SegList<int> sgl;  dctx->setFileFilter(sgl);  CPPUNIT_ASSERT(dctx->getFileEntries()[0]->isRequested());  CPPUNIT_ASSERT(dctx->getFileEntries()[1]->isRequested());  sgl.add(2, 3);  dctx->setFileFilter(sgl);  CPPUNIT_ASSERT(!dctx->getFileEntries()[0]->isRequested());  CPPUNIT_ASSERT(dctx->getFileEntries()[1]->isRequested());  sgl.clear();  sgl.add(3, 4);  dctx->setFileFilter(sgl);  CPPUNIT_ASSERT(!dctx->getFileEntries()[0]->isRequested());  CPPUNIT_ASSERT(!dctx->getFileEntries()[1]->isRequested());  sgl.clear();  util::parseIntSegments(sgl, "1,2");  dctx->setFileFilter(sgl);  CPPUNIT_ASSERT(dctx->getFileEntries()[0]->isRequested());  CPPUNIT_ASSERT(dctx->getFileEntries()[1]->isRequested());}void BittorrentHelperTest::testUTF8Torrent(){  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/utf8.torrent", dctx, option_);  CPPUNIT_ASSERT_EQUAL(std::string("name in utf-8"),                       getTorrentAttrs(dctx)->name);  CPPUNIT_ASSERT_EQUAL(std::string("./name in utf-8/path in utf-8"),                       dctx->getFirstFileEntry()->getPath());  CPPUNIT_ASSERT_EQUAL(std::string("This is utf8 comment."),                       getTorrentAttrs(dctx)->comment);}void BittorrentHelperTest::testEtc(){  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/test.torrent", dctx, option_);  CPPUNIT_ASSERT_EQUAL(std::string("REDNOAH.COM RULES"),                       getTorrentAttrs(dctx)->comment);  CPPUNIT_ASSERT_EQUAL(std::string("aria2"),                       getTorrentAttrs(dctx)->createdBy);  CPPUNIT_ASSERT_EQUAL((time_t)1123456789,                       getTorrentAttrs(dctx)->creationDate);}void BittorrentHelperTest::testCheckBitfield(){  unsigned char bitfield[] = { 0xff, 0xe0 };  checkBitfield(bitfield, sizeof(bitfield), 11);  try {    checkBitfield(bitfield, sizeof(bitfield), 17);    CPPUNIT_FAIL("exception must be thrown.");  } catch(RecoverableException& e) {    // success  }  // Change last byte  bitfield[1] = 0xf0;  try {    checkBitfield(bitfield, sizeof(bitfield), 11);    CPPUNIT_FAIL("exception must be thrown.");  } catch(RecoverableException& e) {    // success  }}void BittorrentHelperTest::testMetadata() {  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/test.torrent", dctx, option_);  std::string torrentData = readFile(A2_TEST_DIR"/test.torrent");  SharedHandle<ValueBase> tr = bencode2::decode(torrentData);  SharedHandle<ValueBase> infoDic = downcast<Dict>(tr)->get("info");  std::string metadata = bencode2::encode(infoDic);  SharedHandle<TorrentAttribute> attrs = getTorrentAttrs(dctx);  CPPUNIT_ASSERT(metadata == attrs->metadata);  CPPUNIT_ASSERT_EQUAL(metadata.size(), attrs->metadataSize);}void BittorrentHelperTest::testParseMagnet(){  std::string magnet =    "magnet:?xt=urn:btih:248d0a1cd08284299de78d5c1ed359bb46717d8c&dn=aria2"    "&tr=http://tracker1&tr=http://tracker2";  SharedHandle<TorrentAttribute> attrs = bittorrent::parseMagnet(magnet);  CPPUNIT_ASSERT_EQUAL(std::string("248d0a1cd08284299de78d5c1ed359bb46717d8c"),                       util::toHex(attrs->infoHash));  CPPUNIT_ASSERT_EQUAL(std::string("[METADATA]aria2"), attrs->name);  CPPUNIT_ASSERT_EQUAL((size_t)2, attrs->announceList.size());  CPPUNIT_ASSERT_EQUAL(std::string("http://tracker1"),                       attrs->announceList[0][0]);  CPPUNIT_ASSERT_EQUAL(std::string("http://tracker2"),                       attrs->announceList[1][0]);  magnet = "magnet:?xt=urn:btih:248d0a1cd08284299de78d5c1ed359bb46717d8c";  attrs = bittorrent::parseMagnet(magnet);  CPPUNIT_ASSERT_EQUAL    (std::string("[METADATA]248d0a1cd08284299de78d5c1ed359bb46717d8c"),     attrs->name);  CPPUNIT_ASSERT(attrs->announceList.empty());  magnet = "magnet:?xt=urn:sha1:7899bdb90a026c746f3cbc10839dd9b2a2a3e985&"    "xt=urn:btih:248d0a1cd08284299de78d5c1ed359bb46717d8c";  attrs = bittorrent::parseMagnet(magnet);  CPPUNIT_ASSERT_EQUAL(std::string("248d0a1cd08284299de78d5c1ed359bb46717d8c"),                       util::toHex(attrs->infoHash));}void BittorrentHelperTest::testParseMagnet_base32(){  std::string infoHash = "248d0a1cd08284299de78d5c1ed359bb46717d8c";  std::string base32InfoHash = base32::encode(fromHex(infoHash));  std::string magnet = "magnet:?xt=urn:btih:"+base32InfoHash+"&dn=aria2";  SharedHandle<TorrentAttribute> attrs = bittorrent::parseMagnet(magnet);  CPPUNIT_ASSERT_EQUAL    (std::string("248d0a1cd08284299de78d5c1ed359bb46717d8c"),     util::toHex(attrs->infoHash));}void BittorrentHelperTest::testMetadata2Torrent(){  std::string metadata = "METADATA";  SharedHandle<TorrentAttribute> attrs(new TorrentAttribute());  CPPUNIT_ASSERT_EQUAL    (std::string("d4:infoMETADATAe"), metadata2Torrent(metadata, attrs));  attrs->announceList.push_back(std::vector<std::string>());  attrs->announceList[0].push_back("http://localhost/announce");  CPPUNIT_ASSERT_EQUAL    (std::string("d"                 "13:announce-list"                 "ll25:http://localhost/announceee"                 "4:infoMETADATA"                 "e"),     metadata2Torrent(metadata, attrs));}void BittorrentHelperTest::testTorrent2Magnet(){  SharedHandle<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR"/test.torrent", dctx, option_);  CPPUNIT_ASSERT_EQUAL    (std::string("magnet:?xt=urn:btih:248D0A1CD08284299DE78D5C1ED359BB46717D8C"                 "&dn=aria2-test"                 "&tr=http%3A%2F%2Ftracker1"                 "&tr=http%3A%2F%2Ftracker2"                 "&tr=http%3A%2F%2Ftracker3"),     torrent2Magnet(getTorrentAttrs(dctx)));}void BittorrentHelperTest::testExtractPeerFromString(){  std::string hextext = "100210354527354678541237324732171ae1";  hextext += "20010db8bd0501d2288a1fc0000110ee1ae2";  std::string peersstr = "36:"+fromHex(hextext);  SharedHandle<ValueBase> str = bencode2::decode(peersstr);  std::deque<SharedHandle<Peer> > peers;  extractPeer(str, AF_INET6, std::back_inserter(peers));  CPPUNIT_ASSERT_EQUAL((size_t)2, peers.size());  CPPUNIT_ASSERT_EQUAL(std::string("1002:1035:4527:3546:7854:1237:3247:3217"),                       peers[0]->getIPAddress());  CPPUNIT_ASSERT_EQUAL((uint16_t)6881, peers[0]->getPort());  CPPUNIT_ASSERT_EQUAL(std::string("2001:db8:bd05:1d2:288a:1fc0:1:10ee"),                       peers[1]->getIPAddress());  CPPUNIT_ASSERT_EQUAL((uint16_t)6882, peers[1]->getPort());  hextext = "c0a800011ae1";  hextext += "c0a800021ae2";  peersstr = "12:"+fromHex(hextext);  str = bencode2::decode(peersstr);  peers.clear();  extractPeer(str, AF_INET, std::back_inserter(peers));  CPPUNIT_ASSERT_EQUAL((size_t)2, peers.size());  CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.1"), peers[0]->getIPAddress());  CPPUNIT_ASSERT_EQUAL((uint16_t)6881, peers[0]->getPort());  CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.2"), peers[1]->getIPAddress());  CPPUNIT_ASSERT_EQUAL((uint16_t)6882, peers[1]->getPort());}void BittorrentHelperTest::testExtractPeerFromList(){  std::string peersString =    "d5:peersld2:ip11:192.168.0.17:peer id20:aria2-00000000000000"    "4:porti2006eeee";  SharedHandle<ValueBase> dict = bencode2::decode(peersString);  std::deque<SharedHandle<Peer> > peers;  extractPeer(downcast<Dict>(dict)->get("peers"), AF_INET, std::back_inserter(peers));  CPPUNIT_ASSERT_EQUAL((size_t)1, peers.size());  SharedHandle<Peer> peer = *peers.begin();  CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.1"), peer->getIPAddress());  CPPUNIT_ASSERT_EQUAL((uint16_t)2006, peer->getPort());}void BittorrentHelperTest::testExtract2PeersFromList(){  std::string peersString =    "d5:peersld2:ip11:192.168.0.17:peer id20:aria2-00000000000000"    "4:porti65535eed2:ip11:192.168.0.27:peer id20:aria2-00000000000000"    "4:porti2007eeee";  SharedHandle<ValueBase> dict = bencode2::decode(peersString);  std::deque<SharedHandle<Peer> > peers;  extractPeer(downcast<Dict>(dict)->get("peers"), AF_INET, std::back_inserter(peers));  CPPUNIT_ASSERT_EQUAL((size_t)2, peers.size());  SharedHandle<Peer> peer = *peers.begin();  CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.1"), peer->getIPAddress());  CPPUNIT_ASSERT_EQUAL((uint16_t)65535, peer->getPort());  peer = *(peers.begin()+1);  CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.2"), peer->getIPAddress());  CPPUNIT_ASSERT_EQUAL((uint16_t)2007, peer->getPort());}void BittorrentHelperTest::testPackcompact(){  unsigned char compact[COMPACT_LEN_IPV6];  CPPUNIT_ASSERT_EQUAL(18,                       packcompact(compact,                                   "1002:1035:4527:3546:7854:1237:3247:3217",                                   6881));  CPPUNIT_ASSERT_EQUAL(std::string("100210354527354678541237324732171ae1"),                       util::toHex(compact, 18));  CPPUNIT_ASSERT_EQUAL(6, packcompact(compact, "192.168.0.1", 6881));  CPPUNIT_ASSERT_EQUAL(std::string("c0a800011ae1"), util::toHex(compact, 6));  CPPUNIT_ASSERT_EQUAL(0, packcompact(compact, "badaddr", 6881));}void BittorrentHelperTest::testUnpackcompact(){  unsigned char compact6[] = {    0x10, 0x02, 0x10, 0x35, 0x45, 0x27, 0x35, 0x46,    0x78, 0x54, 0x12, 0x37, 0x32, 0x47, 0x32, 0x17,    0x1A, 0xE1 };  std::pair<std::string, uint16_t> p =    unpackcompact(compact6, AF_INET6);  CPPUNIT_ASSERT_EQUAL(std::string("1002:1035:4527:3546:7854:1237:3247:3217"),                       p.first);  CPPUNIT_ASSERT_EQUAL((uint16_t)6881, p.second);  unsigned char compact[] = { 0xC0, 0xa8, 0x00, 0x01, 0x1A, 0xE1 };  p = unpackcompact(compact, AF_INET);  CPPUNIT_ASSERT_EQUAL(std::string("192.168.0.1"), p.first);  CPPUNIT_ASSERT_EQUAL((uint16_t)6881, p.second);}void BittorrentHelperTest::testRemoveAnnounceUri(){  SharedHandle<TorrentAttribute> attrs(new TorrentAttribute());  std::vector<std::string> tier1;  tier1.push_back("http://host1/announce");  std::vector<std::string> tier2;  tier2.push_back("http://host2/announce");  tier2.push_back("http://host3/announce");  attrs->announceList.push_back(tier1);  attrs->announceList.push_back(tier2);  std::vector<std::string> removeUris;  removeUris.push_back(tier1[0]);  removeUris.push_back(tier2[0]);  removeAnnounceUri(attrs, removeUris);  CPPUNIT_ASSERT_EQUAL((size_t)1, attrs->announceList.size());  CPPUNIT_ASSERT_EQUAL(std::string("http://host3/announce"),                       attrs->announceList[0][0]);  removeUris.clear();  removeUris.push_back("*");  removeAnnounceUri(attrs, removeUris);  CPPUNIT_ASSERT(attrs->announceList.empty());}void BittorrentHelperTest::testAddAnnounceUri(){  SharedHandle<TorrentAttribute> attrs(new TorrentAttribute());  std::vector<std::string> addUris;  addUris.push_back("http://host1/announce");  addUris.push_back("http://host2/announce");  addAnnounceUri(attrs, addUris);  CPPUNIT_ASSERT_EQUAL((size_t)2, attrs->announceList.size());  CPPUNIT_ASSERT_EQUAL((size_t)1, attrs->announceList[0].size());  CPPUNIT_ASSERT_EQUAL(std::string("http://host1/announce"),                       attrs->announceList[0][0]);  CPPUNIT_ASSERT_EQUAL((size_t)1, attrs->announceList[1].size());  CPPUNIT_ASSERT_EQUAL(std::string("http://host2/announce"),                       attrs->announceList[1][0]);}void BittorrentHelperTest::testAdjustAnnounceUri(){  SharedHandle<TorrentAttribute> attrs(new TorrentAttribute());  std::vector<std::string> tier1;  tier1.push_back("http://host1/announce");  std::vector<std::string> tier2;  tier2.push_back("http://host2/announce");  tier2.push_back("http://host3/announce");  attrs->announceList.push_back(tier1);  attrs->announceList.push_back(tier2);  SharedHandle<Option> option(new Option());  option->put(PREF_BT_TRACKER, "http://host1/announce,http://host4/announce");  option->put(PREF_BT_EXCLUDE_TRACKER,              "http://host1/announce,http://host2/announce");  adjustAnnounceUri(attrs, option);  CPPUNIT_ASSERT_EQUAL((size_t)3, attrs->announceList.size());  CPPUNIT_ASSERT_EQUAL((size_t)1, attrs->announceList[0].size());  CPPUNIT_ASSERT_EQUAL(std::string("http://host3/announce"),                       attrs->announceList[0][0]);  CPPUNIT_ASSERT_EQUAL((size_t)1, attrs->announceList[1].size());  CPPUNIT_ASSERT_EQUAL(std::string("http://host1/announce"),                       attrs->announceList[1][0]);  CPPUNIT_ASSERT_EQUAL((size_t)1, attrs->announceList[2].size());  CPPUNIT_ASSERT_EQUAL(std::string("http://host4/announce"),                       attrs->announceList[2][0]);}} // namespace bittorrent} // namespace aria2
 |