| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036 | #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:  std::shared_ptr<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(){  std::shared_ptr<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(){  std::shared_ptr<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(){  std::shared_ptr<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<std::shared_ptr<FileEntry>> fileEntries = dctx->getFileEntries();  // There are 2 file entries.  CPPUNIT_ASSERT_EQUAL((size_t)2, fileEntries.size());  std::vector<std::shared_ptr<FileEntry>>::iterator itr = fileEntries.begin();  std::shared_ptr<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++;  std::shared_ptr<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(){  std::shared_ptr<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<std::shared_ptr<FileEntry>> fileEntries = dctx->getFileEntries();  // There is 1 file entry.  CPPUNIT_ASSERT_EQUAL((size_t)1, fileEntries.size());  std::vector<std::shared_ptr<FileEntry>>::iterator itr = fileEntries.begin();  std::shared_ptr<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(std::string("aria2-0.8.2.tar.bz2"),                       fileEntry1->getSuffixPath());  CPPUNIT_ASSERT_EQUAL(10, fileEntry1->getMaxConnectionPerServer());}void BittorrentHelperTest::testGetTotalLength(){  std::shared_ptr<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR "/test.torrent", dctx, option_);  CPPUNIT_ASSERT_EQUAL((int64_t)384LL, dctx->getTotalLength());}void BittorrentHelperTest::testGetTotalLengthSingle(){  std::shared_ptr<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR "/single.torrent", dctx, option_);  CPPUNIT_ASSERT_EQUAL((int64_t)384LL, dctx->getTotalLength());}void BittorrentHelperTest::testGetFileModeMulti(){  std::shared_ptr<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR "/test.torrent", dctx, option_);  CPPUNIT_ASSERT_EQUAL(BT_FILE_MODE_MULTI, getTorrentAttrs(dctx)->mode);}void BittorrentHelperTest::testGetFileModeSingle(){  std::shared_ptr<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR "/single.torrent", dctx, option_);  CPPUNIT_ASSERT_EQUAL(BT_FILE_MODE_SINGLE, getTorrentAttrs(dctx)->mode);}void BittorrentHelperTest::testGetNameMulti(){  std::shared_ptr<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(){  std::shared_ptr<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(){  std::shared_ptr<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(){  std::shared_ptr<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR "/single.torrent", dctx, option_);  auto 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(){  std::shared_ptr<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR "/test.torrent", dctx, option_);  auto 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(){  std::shared_ptr<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR "/test.torrent", dctx, option_);  CPPUNIT_ASSERT_EQUAL(128, dctx->getPieceLength());}void BittorrentHelperTest::testGetInfoHashAsString(){  std::shared_ptr<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;  {    auto fastSet = computeFastSet(ipaddr, numPieces, infoHash, fastSetSize);    size_t ans[] = {686, 459, 278, 200, 404, 834, 64, 203, 760, 950};    CPPUNIT_ASSERT(std::equal(fastSet.begin(), fastSet.end(), std::begin(ans)));  }  ipaddr = "10.0.0.1";  {    auto fastSet = computeFastSet(ipaddr, numPieces, infoHash, fastSetSize);    size_t ans[] = {568, 188, 466, 452, 550, 662, 109, 226, 398, 11};    CPPUNIT_ASSERT(std::equal(fastSet.begin(), fastSet.end(), std::begin(ans)));  }  // See when pieces < fastSetSize  numPieces = 9;  {    auto fastSet = computeFastSet(ipaddr, numPieces, infoHash, fastSetSize);    size_t ans[] = {8, 6, 7, 5, 1, 4, 0, 2, 3};    CPPUNIT_ASSERT(std::equal(fastSet.begin(), fastSet.end(), std::begin(ans)));  }}void BittorrentHelperTest::testGetFileEntries_multiFileUrlList(){  std::shared_ptr<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR "/url-list-multiFile.torrent", dctx, option_);  // This is multi-file torrent.  const std::vector<std::shared_ptr<FileEntry>>& fileEntries =      dctx->getFileEntries();  // There are 2 file entries.  CPPUNIT_ASSERT_EQUAL((size_t)2, fileEntries.size());  std::vector<std::shared_ptr<FileEntry>>::const_iterator itr =      fileEntries.begin();  const std::shared_ptr<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->getSuffixPath());  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 std::shared_ptr<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(){  std::shared_ptr<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR "/url-list-singleFile.torrent", dctx, option_);  // This is single-file torrent.  const std::vector<std::shared_ptr<FileEntry>>& fileEntries =      dctx->getFileEntries();  // There are 1 file entries.  CPPUNIT_ASSERT_EQUAL((size_t)1, fileEntries.size());  const std::shared_ptr<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(){  std::shared_ptr<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR "/url-list-singleFileEndsWithSlash.torrent", dctx, option_);  // This is single-file torrent.  const std::vector<std::shared_ptr<FileEntry>>& fileEntries =      dctx->getFileEntries();  // There are 1 file entries.  CPPUNIT_ASSERT_EQUAL((size_t)1, fileEntries.size());  const std::shared_ptr<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(){  auto path = List::g();  path->append("path");  path->append(fromHex("90a28a") + "E");  auto file = Dict::g();  file->put("length", Integer::g(1_k));  file->put("path", std::move(path));  auto files = List::g();  files->append(std::move(file));  auto info = Dict::g();  info->put("files", std::move(files));  info->put("piece length", Integer::g(1_k));  info->put("pieces", "01234567890123456789");  info->put("name", fromHex("1b") + "$B%O%m!<" + fromHex("1b") + "(B");  Dict dict;  dict.put("info", std::move(info));  auto dctx = std::make_shared<DownloadContext>();  loadFromMemory(bencode2::encode(&dict), dctx, option_, "default");  auto& 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/path/%90%A2%8AE"),      fe->getSuffixPath());  CPPUNIT_ASSERT_EQUAL(std::string("./%1B%24B%25O%25m%21%3C%1B%28B"),                       dctx->getBasePath());}void BittorrentHelperTest::testLoadFromMemory_singleFileNonUtf8Path(){  auto info = Dict::g();  info->put("piece length", Integer::g(1_k));  info->put("pieces", "01234567890123456789");  info->put("name", fromHex("90a28a") + "E");  info->put("length", Integer::g(1_k));  Dict dict;  dict.put("info", std::move(info));  auto dctx = std::make_shared<DownloadContext>();  loadFromMemory(bencode2::encode(&dict), dctx, option_, "default");  const std::shared_ptr<FileEntry>& fe = dctx->getFirstFileEntry();  CPPUNIT_ASSERT_EQUAL(std::string("./%90%A2%8AE"), fe->getPath());  CPPUNIT_ASSERT_EQUAL(std::string("%90%A2%8AE"), fe->getSuffixPath());}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:"                       "AAAAAAAAAAAAAAAAAAAABBBBBBBBBBBBBBBBBBBBCCCCCCCCCCCCCCC"                       "CCCCCee";  std::shared_ptr<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";    std::shared_ptr<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:"                       "AAAAAAAAAAAAAAAAAAAABBBBBBBBBBBBBBBBBBBBCCCCCCCCCCCCCCC"                       "CCCCCee";  std::shared_ptr<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";  std::shared_ptr<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";  std::shared_ptr<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";    std::shared_ptr<DownloadContext> dctx(new DownloadContext());    loadFromMemory(memory, dctx, option_, "default");    auto 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";    std::shared_ptr<DownloadContext> dctx(new DownloadContext());    loadFromMemory(memory, dctx, option_, "default");    auto 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";    std::shared_ptr<DownloadContext> dctx(new DownloadContext());    loadFromMemory(memory, dctx, option_, "default");    auto 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";    std::shared_ptr<DownloadContext> dctx(new DownloadContext());    loadFromMemory(memory, dctx, option_, "default");    auto 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";    std::shared_ptr<DownloadContext> dctx(new DownloadContext());    loadFromMemory(memory, dctx, option_, "default");    auto 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";    std::shared_ptr<DownloadContext> dctx(new DownloadContext());    loadFromMemory(memory, dctx, option_, "default");    auto 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(){  std::shared_ptr<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");  std::shared_ptr<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(){  std::shared_ptr<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR "/single.torrent", dctx, option_);  CPPUNIT_ASSERT(dctx->getFirstFileEntry()->isRequested());  dctx->setFileFilter(SegList<int>());  CPPUNIT_ASSERT(dctx->getFirstFileEntry()->isRequested());  dctx->setFileFilter(util::parseIntSegments("1,2"));  CPPUNIT_ASSERT(dctx->getFirstFileEntry()->isRequested());  // For single file torrent, file is always selected whatever range  // is passed.  dctx->setFileFilter(util::parseIntSegments("2,3"));  CPPUNIT_ASSERT(dctx->getFirstFileEntry()->isRequested());}void BittorrentHelperTest::testSetFileFilter_multi(){  std::shared_ptr<DownloadContext> dctx(new DownloadContext());  load(A2_TEST_DIR "/test.torrent", dctx, option_);  CPPUNIT_ASSERT(dctx->getFileEntries()[0]->isRequested());  CPPUNIT_ASSERT(dctx->getFileEntries()[1]->isRequested());  dctx->setFileFilter(SegList<int>());  CPPUNIT_ASSERT(dctx->getFileEntries()[0]->isRequested());  CPPUNIT_ASSERT(dctx->getFileEntries()[1]->isRequested());  dctx->setFileFilter(util::parseIntSegments("2,3"));  CPPUNIT_ASSERT(!dctx->getFileEntries()[0]->isRequested());  CPPUNIT_ASSERT(dctx->getFileEntries()[1]->isRequested());  dctx->setFileFilter(util::parseIntSegments("3,4"));  CPPUNIT_ASSERT(!dctx->getFileEntries()[0]->isRequested());  CPPUNIT_ASSERT(!dctx->getFileEntries()[1]->isRequested());  dctx->setFileFilter(util::parseIntSegments("1,2"));  CPPUNIT_ASSERT(dctx->getFileEntries()[0]->isRequested());  CPPUNIT_ASSERT(dctx->getFileEntries()[1]->isRequested());}void BittorrentHelperTest::testUTF8Torrent(){  std::shared_ptr<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(){  std::shared_ptr<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(){  auto dctx = std::make_shared<DownloadContext>();  load(A2_TEST_DIR "/test.torrent", dctx, option_);  std::string torrentData = readFile(A2_TEST_DIR "/test.torrent");  auto tr = bencode2::decode(torrentData);  auto infoDic = downcast<Dict>(tr)->get("info");  std::string metadata = bencode2::encode(infoDic);  auto 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";  auto 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";  auto attrs = bittorrent::parseMagnet(magnet);  CPPUNIT_ASSERT_EQUAL(std::string("248d0a1cd08284299de78d5c1ed359bb46717d8c"),                       util::toHex(attrs->infoHash));}void BittorrentHelperTest::testMetadata2Torrent(){  TorrentAttribute attrs;  std::string metadata = "METADATA";  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(){  std::shared_ptr<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);  auto str = bencode2::decode(peersstr);  std::deque<std::shared_ptr<Peer>> peers;  extractPeer(str.get(), 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.get(), 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";  auto dict = bencode2::decode(peersString);  std::deque<std::shared_ptr<Peer>> peers;  extractPeer(downcast<Dict>(dict)->get("peers"), AF_INET,              std::back_inserter(peers));  CPPUNIT_ASSERT_EQUAL((size_t)1, peers.size());  auto& 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";  auto dict = bencode2::decode(peersString);  std::deque<std::shared_ptr<Peer>> peers;  extractPeer(downcast<Dict>(dict)->get("peers"), AF_INET,              std::back_inserter(peers));  CPPUNIT_ASSERT_EQUAL((size_t)2, peers.size());  auto& 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(      (size_t)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((size_t)6, packcompact(compact, "192.168.0.1", 6881));  CPPUNIT_ASSERT_EQUAL(std::string("c0a800011ae1"), util::toHex(compact, 6));  CPPUNIT_ASSERT_EQUAL((size_t)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(){  TorrentAttribute attrs;  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(){  TorrentAttribute attrs;  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(){  TorrentAttribute attrs;  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::shared_ptr<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
 |