| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515 | #include "DHTMessageFactoryImpl.h"#include <cstring>#include <iostream>#include <cppunit/extensions/HelperMacros.h>#include "RecoverableException.h"#include "util.h"#include "DHTNode.h"#include "DHTRoutingTable.h"#include "Peer.h"#include "bittorrent_helper.h"#include "DHTBucket.h"#include "DHTPingMessage.h"#include "DHTPingReplyMessage.h"#include "DHTFindNodeMessage.h"#include "DHTFindNodeReplyMessage.h"#include "DHTGetPeersMessage.h"#include "DHTGetPeersReplyMessage.h"#include "DHTAnnouncePeerMessage.h"#include "DHTAnnouncePeerReplyMessage.h"#include "bencode2.h"namespace aria2 {class DHTMessageFactoryImplTest:public CppUnit::TestFixture {  CPPUNIT_TEST_SUITE(DHTMessageFactoryImplTest);  CPPUNIT_TEST(testCreatePingMessage);  CPPUNIT_TEST(testCreatePingReplyMessage);  CPPUNIT_TEST(testCreateFindNodeMessage);  CPPUNIT_TEST(testCreateFindNodeReplyMessage);  CPPUNIT_TEST(testCreateFindNodeReplyMessage6);  CPPUNIT_TEST(testCreateGetPeersMessage);  CPPUNIT_TEST(testCreateGetPeersReplyMessage);  CPPUNIT_TEST(testCreateGetPeersReplyMessage6);  CPPUNIT_TEST(testCreateAnnouncePeerMessage);  CPPUNIT_TEST(testCreateAnnouncePeerReplyMessage);  CPPUNIT_TEST(testReceivedErrorMessage);  CPPUNIT_TEST_SUITE_END();public:  SharedHandle<DHTMessageFactoryImpl> factory;  SharedHandle<DHTRoutingTable> routingTable;  SharedHandle<DHTNode> localNode;  unsigned char transactionID[DHT_TRANSACTION_ID_LENGTH];  unsigned char remoteNodeID[DHT_ID_LENGTH];  void setUp()  {    localNode.reset(new DHTNode());    factory.reset(new DHTMessageFactoryImpl(AF_INET));    factory->setLocalNode(localNode);    memset(transactionID, 0xff, DHT_TRANSACTION_ID_LENGTH);    memset(remoteNodeID, 0x0f, DHT_ID_LENGTH);    routingTable.reset(new DHTRoutingTable(localNode));    factory->setRoutingTable(routingTable.get());  }  void tearDown() {}  void testCreatePingMessage();  void testCreatePingReplyMessage();  void testCreateFindNodeMessage();  void testCreateFindNodeReplyMessage();  void testCreateFindNodeReplyMessage6();  void testCreateGetPeersMessage();  void testCreateGetPeersReplyMessage();  void testCreateGetPeersReplyMessage6();  void testCreateAnnouncePeerMessage();  void testCreateAnnouncePeerReplyMessage();  void testReceivedErrorMessage();};CPPUNIT_TEST_SUITE_REGISTRATION(DHTMessageFactoryImplTest);void DHTMessageFactoryImplTest::testCreatePingMessage(){  Dict dict;  dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));  dict.put("y", "q");  dict.put("q", "ping");  SharedHandle<Dict> aDict = Dict::g();  aDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));  dict.put("a", aDict);    SharedHandle<DHTPingMessage> m    (dynamic_pointer_cast<DHTPingMessage>     (factory->createQueryMessage(&dict, "192.168.0.1", 6881)));  SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));  remoteNode->setIPAddress("192.168.0.1");  remoteNode->setPort(6881);  CPPUNIT_ASSERT(*localNode == *m->getLocalNode());  CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());  CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),                       util::toHex(m->getTransactionID()));}void DHTMessageFactoryImplTest::testCreatePingReplyMessage(){  Dict dict;  dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));  dict.put("y", "r");  SharedHandle<Dict> rDict = Dict::g();  rDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));  dict.put("r", rDict);  SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));  remoteNode->setIPAddress("192.168.0.1");  remoteNode->setPort(6881);    SharedHandle<DHTPingReplyMessage> m    (dynamic_pointer_cast<DHTPingReplyMessage>     (factory->createResponseMessage("ping", &dict,                                     remoteNode->getIPAddress(),                                     remoteNode->getPort())));  CPPUNIT_ASSERT(*localNode == *m->getLocalNode());  CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());  CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),                       util::toHex(m->getTransactionID()));}void DHTMessageFactoryImplTest::testCreateFindNodeMessage(){  Dict dict;  dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));  dict.put("y", "q");  dict.put("q", "find_node");  SharedHandle<Dict> aDict = Dict::g();  aDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));  unsigned char targetNodeID[DHT_ID_LENGTH];  memset(targetNodeID, 0x11, DHT_ID_LENGTH);  aDict->put("target", String::g(targetNodeID, DHT_ID_LENGTH));  dict.put("a", aDict);    SharedHandle<DHTFindNodeMessage> m    (dynamic_pointer_cast<DHTFindNodeMessage>     (factory->createQueryMessage(&dict, "192.168.0.1", 6881)));  SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));  remoteNode->setIPAddress("192.168.0.1");  remoteNode->setPort(6881);  CPPUNIT_ASSERT(*localNode == *m->getLocalNode());  CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());  CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),                       util::toHex(m->getTransactionID()));  CPPUNIT_ASSERT_EQUAL(util::toHex(targetNodeID, DHT_ID_LENGTH),                       util::toHex(m->getTargetNodeID(), DHT_ID_LENGTH));}void DHTMessageFactoryImplTest::testCreateFindNodeReplyMessage(){  try {    Dict dict;    dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));    dict.put("y", "r");    SharedHandle<Dict> rDict = Dict::g();    rDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));    std::string compactNodeInfo;    SharedHandle<DHTNode> nodes[8];    for(size_t i = 0; i < DHTBucket::K; ++i) {      nodes[i].reset(new DHTNode());      nodes[i]->setIPAddress("192.168.0."+util::uitos(i+1));      nodes[i]->setPort(6881+i);      unsigned char buf[COMPACT_LEN_IPV6];      CPPUNIT_ASSERT_EQUAL        (COMPACT_LEN_IPV4,         bittorrent::packcompact         (buf, nodes[i]->getIPAddress(), nodes[i]->getPort()));      compactNodeInfo +=        std::string(&nodes[i]->getID()[0], &nodes[i]->getID()[DHT_ID_LENGTH])+        std::string(&buf[0], &buf[COMPACT_LEN_IPV4]);    }    rDict->put("nodes", compactNodeInfo);    dict.put("r", rDict);    SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));    remoteNode->setIPAddress("192.168.0.1");    remoteNode->setPort(6881);      SharedHandle<DHTFindNodeReplyMessage> m      (dynamic_pointer_cast<DHTFindNodeReplyMessage>       (factory->createResponseMessage("find_node", &dict,                                       remoteNode->getIPAddress(),                                       remoteNode->getPort())));    CPPUNIT_ASSERT(*localNode == *m->getLocalNode());    CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());    CPPUNIT_ASSERT_EQUAL((size_t)DHTBucket::K, m->getClosestKNodes().size());    CPPUNIT_ASSERT(*nodes[0] == *m->getClosestKNodes()[0]);    CPPUNIT_ASSERT(*nodes[7] == *m->getClosestKNodes()[7]);    CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),                         util::toHex(m->getTransactionID()));  } catch(Exception& e) {    CPPUNIT_FAIL(e.stackTrace());  }}void DHTMessageFactoryImplTest::testCreateFindNodeReplyMessage6(){  factory.reset(new DHTMessageFactoryImpl(AF_INET6));  factory->setLocalNode(localNode);  factory->setRoutingTable(routingTable.get());  try {    Dict dict;    dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));    dict.put("y", "r");    SharedHandle<Dict> rDict = Dict::g();    rDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));    std::string compactNodeInfo;    SharedHandle<DHTNode> nodes[8];    for(size_t i = 0; i < DHTBucket::K; ++i) {      nodes[i].reset(new DHTNode());      nodes[i]->setIPAddress("2001::000"+util::uitos(i+1));      nodes[i]->setPort(6881+i);      unsigned char buf[COMPACT_LEN_IPV6];      CPPUNIT_ASSERT_EQUAL        (COMPACT_LEN_IPV6,         bittorrent::packcompact         (buf, nodes[i]->getIPAddress(), nodes[i]->getPort()));      compactNodeInfo +=        std::string(&nodes[i]->getID()[0], &nodes[i]->getID()[DHT_ID_LENGTH])+        std::string(&buf[0], &buf[COMPACT_LEN_IPV6]);    }    rDict->put("nodes6", compactNodeInfo);    dict.put("r", rDict);    SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));    remoteNode->setIPAddress("2001::2001");    remoteNode->setPort(6881);      SharedHandle<DHTFindNodeReplyMessage> m      (dynamic_pointer_cast<DHTFindNodeReplyMessage>       (factory->createResponseMessage("find_node", &dict,                                       remoteNode->getIPAddress(),                                       remoteNode->getPort())));    CPPUNIT_ASSERT(*localNode == *m->getLocalNode());    CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());    CPPUNIT_ASSERT_EQUAL((size_t)DHTBucket::K, m->getClosestKNodes().size());    CPPUNIT_ASSERT(*nodes[0] == *m->getClosestKNodes()[0]);    CPPUNIT_ASSERT(*nodes[7] == *m->getClosestKNodes()[7]);    CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),                         util::toHex(m->getTransactionID()));  } catch(Exception& e) {    CPPUNIT_FAIL(e.stackTrace());  }}void DHTMessageFactoryImplTest::testCreateGetPeersMessage(){  Dict dict;  dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));  dict.put("y", "q");  dict.put("q", "get_peers");  SharedHandle<Dict> aDict = Dict::g();  aDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));  unsigned char infoHash[DHT_ID_LENGTH];  memset(infoHash, 0x11, DHT_ID_LENGTH);  aDict->put("info_hash", String::g(infoHash, DHT_ID_LENGTH));  dict.put("a", aDict);    SharedHandle<DHTGetPeersMessage> m    (dynamic_pointer_cast<DHTGetPeersMessage>     (factory->createQueryMessage(&dict, "192.168.0.1", 6881)));  SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));  remoteNode->setIPAddress("192.168.0.1");  remoteNode->setPort(6881);  CPPUNIT_ASSERT(*localNode == *m->getLocalNode());  CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());  CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),                       util::toHex(m->getTransactionID()));  CPPUNIT_ASSERT_EQUAL(util::toHex(infoHash, DHT_ID_LENGTH),                       util::toHex(m->getInfoHash(), DHT_ID_LENGTH));}void DHTMessageFactoryImplTest::testCreateGetPeersReplyMessage(){  try {    Dict dict;    dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));    dict.put("y", "r");    SharedHandle<Dict> rDict = Dict::g();    rDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));    std::string compactNodeInfo;    SharedHandle<DHTNode> nodes[8];    for(size_t i = 0; i < DHTBucket::K; ++i) {      nodes[i].reset(new DHTNode());      nodes[i]->setIPAddress("192.168.0."+util::uitos(i+1));      nodes[i]->setPort(6881+i);      unsigned char buf[COMPACT_LEN_IPV6];      CPPUNIT_ASSERT_EQUAL        (COMPACT_LEN_IPV4,         bittorrent::packcompact         (buf, nodes[i]->getIPAddress(), nodes[i]->getPort()));      compactNodeInfo +=        std::string(&nodes[i]->getID()[0], &nodes[i]->getID()[DHT_ID_LENGTH])+        std::string(&buf[0], &buf[COMPACT_LEN_IPV4]);    }    rDict->put("nodes", compactNodeInfo);    std::deque<SharedHandle<Peer> > peers;    SharedHandle<List> valuesList = List::g();    for(size_t i = 0; i < 4; ++i) {      SharedHandle<Peer> peer(new Peer("192.168.0."+util::uitos(i+1), 6881+i));      unsigned char buffer[COMPACT_LEN_IPV6];      CPPUNIT_ASSERT_EQUAL        (COMPACT_LEN_IPV4,         bittorrent::packcompact         (buffer, peer->getIPAddress(), peer->getPort()));      valuesList->append(String::g(buffer, COMPACT_LEN_IPV4));      peers.push_back(peer);    }    rDict->put("values", valuesList);    rDict->put("token", "token");    dict.put("r", rDict);    SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));    remoteNode->setIPAddress("192.168.0.1");    remoteNode->setPort(6881);      SharedHandle<DHTGetPeersReplyMessage> m      (dynamic_pointer_cast<DHTGetPeersReplyMessage>       (factory->createResponseMessage("get_peers", &dict,                                       remoteNode->getIPAddress(),                                       remoteNode->getPort())));    CPPUNIT_ASSERT(*localNode == *m->getLocalNode());    CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());    CPPUNIT_ASSERT_EQUAL(std::string("token"), m->getToken());    CPPUNIT_ASSERT_EQUAL((size_t)DHTBucket::K, m->getClosestKNodes().size());    CPPUNIT_ASSERT(*nodes[0] == *m->getClosestKNodes()[0]);    CPPUNIT_ASSERT(*nodes[7] == *m->getClosestKNodes()[7]);    CPPUNIT_ASSERT_EQUAL((size_t)4, m->getValues().size());    CPPUNIT_ASSERT(*peers[0] == *m->getValues()[0]);    CPPUNIT_ASSERT(*peers[3] == *m->getValues()[3]);    CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),                         util::toHex(m->getTransactionID()));  } catch(Exception& e) {    CPPUNIT_FAIL(e.stackTrace());  }}void DHTMessageFactoryImplTest::testCreateGetPeersReplyMessage6(){  factory.reset(new DHTMessageFactoryImpl(AF_INET6));  factory->setLocalNode(localNode);  factory->setRoutingTable(routingTable.get());  try {    Dict dict;    dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));    dict.put("y", "r");    SharedHandle<Dict> rDict = Dict::g();    rDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));    std::string compactNodeInfo;    SharedHandle<DHTNode> nodes[8];    for(size_t i = 0; i < DHTBucket::K; ++i) {      nodes[i].reset(new DHTNode());      nodes[i]->setIPAddress("2001::000"+util::uitos(i+1));      nodes[i]->setPort(6881+i);      unsigned char buf[COMPACT_LEN_IPV6];      CPPUNIT_ASSERT_EQUAL        (COMPACT_LEN_IPV6,         bittorrent::packcompact         (buf, nodes[i]->getIPAddress(), nodes[i]->getPort()));      compactNodeInfo +=        std::string(&nodes[i]->getID()[0], &nodes[i]->getID()[DHT_ID_LENGTH])+        std::string(&buf[0], &buf[COMPACT_LEN_IPV6]);    }    rDict->put("nodes6", compactNodeInfo);    std::deque<SharedHandle<Peer> > peers;    SharedHandle<List> valuesList = List::g();    for(size_t i = 0; i < 4; ++i) {      SharedHandle<Peer> peer(new Peer("2001::100"+util::uitos(i+1), 6881+i));      unsigned char buffer[COMPACT_LEN_IPV6];      CPPUNIT_ASSERT_EQUAL        (COMPACT_LEN_IPV6,         bittorrent::packcompact         (buffer, peer->getIPAddress(), peer->getPort()));      valuesList->append(String::g(buffer, COMPACT_LEN_IPV6));      peers.push_back(peer);    }    rDict->put("values", valuesList);    rDict->put("token", "token");    dict.put("r", rDict);    SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));    remoteNode->setIPAddress("2001::2001");    remoteNode->setPort(6881);      SharedHandle<DHTGetPeersReplyMessage> m      (dynamic_pointer_cast<DHTGetPeersReplyMessage>       (factory->createResponseMessage("get_peers", &dict,                                       remoteNode->getIPAddress(),                                       remoteNode->getPort())));    CPPUNIT_ASSERT(*localNode == *m->getLocalNode());    CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());    CPPUNIT_ASSERT_EQUAL(std::string("token"), m->getToken());    CPPUNIT_ASSERT_EQUAL((size_t)DHTBucket::K, m->getClosestKNodes().size());    CPPUNIT_ASSERT(*nodes[0] == *m->getClosestKNodes()[0]);    CPPUNIT_ASSERT(*nodes[7] == *m->getClosestKNodes()[7]);    CPPUNIT_ASSERT_EQUAL((size_t)4, m->getValues().size());    CPPUNIT_ASSERT(*peers[0] == *m->getValues()[0]);    CPPUNIT_ASSERT(*peers[3] == *m->getValues()[3]);    CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),                         util::toHex(m->getTransactionID()));  } catch(Exception& e) {    CPPUNIT_FAIL(e.stackTrace());  }}void DHTMessageFactoryImplTest::testCreateAnnouncePeerMessage(){  try {    Dict dict;    dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));    dict.put("y", "q");    dict.put("q", "announce_peer");    SharedHandle<Dict> aDict = Dict::g();    aDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));    unsigned char infoHash[DHT_ID_LENGTH];    memset(infoHash, 0x11, DHT_ID_LENGTH);    aDict->put("info_hash", String::g(infoHash, DHT_ID_LENGTH));    std::string token = "ffff";    uint16_t port = 6881;    aDict->put("port", Integer::g(port));    aDict->put("token", token);    dict.put("a", aDict);      SharedHandle<DHTAnnouncePeerMessage> m      (dynamic_pointer_cast<DHTAnnouncePeerMessage>       (factory->createQueryMessage(&dict, "192.168.0.1", 6882)));    SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));    remoteNode->setIPAddress("192.168.0.1");    remoteNode->setPort(6882);    CPPUNIT_ASSERT(*localNode == *m->getLocalNode());    CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());    CPPUNIT_ASSERT_EQUAL(token, m->getToken());    CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),                         util::toHex(m->getTransactionID()));    CPPUNIT_ASSERT_EQUAL(util::toHex(infoHash, DHT_ID_LENGTH),                         util::toHex(m->getInfoHash(), DHT_ID_LENGTH));    CPPUNIT_ASSERT_EQUAL(port, m->getTCPPort());  } catch(Exception& e) {    CPPUNIT_FAIL(e.stackTrace());  }}void DHTMessageFactoryImplTest::testCreateAnnouncePeerReplyMessage(){  Dict dict;  dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));  dict.put("y", "r");  SharedHandle<Dict> rDict = Dict::g();  rDict->put("id", String::g(remoteNodeID, DHT_ID_LENGTH));  dict.put("r", rDict);  SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));  remoteNode->setIPAddress("192.168.0.1");  remoteNode->setPort(6881);    SharedHandle<DHTAnnouncePeerReplyMessage> m    (dynamic_pointer_cast<DHTAnnouncePeerReplyMessage>     (factory->createResponseMessage("announce_peer", &dict,                                     remoteNode->getIPAddress(),                                     remoteNode->getPort())));  CPPUNIT_ASSERT(*localNode == *m->getLocalNode());  CPPUNIT_ASSERT(*remoteNode == *m->getRemoteNode());  CPPUNIT_ASSERT_EQUAL(util::toHex(transactionID, DHT_TRANSACTION_ID_LENGTH),                       util::toHex(m->getTransactionID()));}void DHTMessageFactoryImplTest::testReceivedErrorMessage(){  Dict dict;  dict.put("t", String::g(transactionID, DHT_TRANSACTION_ID_LENGTH));  dict.put("y", "e");  SharedHandle<List> list = List::g();  list->append(Integer::g(404));  list->append("Not found");  dict.put("e", list);  SharedHandle<DHTNode> remoteNode(new DHTNode(remoteNodeID));  remoteNode->setIPAddress("192.168.0.1");  remoteNode->setPort(6881);  try {    factory->createResponseMessage("announce_peer", &dict,                                   remoteNode->getIPAddress(),                                   remoteNode->getPort());    CPPUNIT_FAIL("exception must be thrown.");  } catch(RecoverableException& e) {    std::cerr << e.stackTrace() << std::endl;  }}} // namespace aria2
 |