/* */ #include "DHTEntryPointNameResolveCommand.h" #include "DownloadEngine.h" #include "NameResolver.h" #include "DlAbortEx.h" #include "prefs.h" #include "message.h" #include "util.h" #include "Option.h" #include "DHTNode.h" #include "DHTTaskQueue.h" #include "DHTTaskFactory.h" #include "DHTRoutingTable.h" #include "DHTTask.h" #include "RequestGroupMan.h" #include "Logger.h" #include "LogFactory.h" #include "fmt.h" #include "SocketCore.h" #ifdef ENABLE_ASYNC_DNS #include "AsyncNameResolverMan.h" #endif // ENABLE_ASYNC_DNS namespace aria2 { DHTEntryPointNameResolveCommand::DHTEntryPointNameResolveCommand( cuid_t cuid, DownloadEngine* e, const std::vector>& entryPoints) : Command{cuid}, e_{e}, #ifdef ENABLE_ASYNC_DNS asyncNameResolverMan_{make_unique()}, #endif // ENABLE_ASYNC_DNS taskQueue_{nullptr}, taskFactory_{nullptr}, routingTable_{nullptr}, entryPoints_(std::begin(entryPoints), std::end(entryPoints)), numSuccess_{0}, bootstrapEnabled_{false} { #ifdef ENABLE_ASYNC_DNS configureAsyncNameResolverMan(asyncNameResolverMan_.get(), e_->getOption()); #endif // ENABLE_ASYNC_DNS } DHTEntryPointNameResolveCommand::~DHTEntryPointNameResolveCommand() { #ifdef ENABLE_ASYNC_DNS asyncNameResolverMan_->disableNameResolverCheck(e_, this); #endif // ENABLE_ASYNC_DNS } bool DHTEntryPointNameResolveCommand::execute() { if (e_->getRequestGroupMan()->downloadFinished() || e_->isHaltRequested()) { return true; } try { #ifdef ENABLE_ASYNC_DNS if (e_->getOption()->getAsBool(PREF_ASYNC_DNS)) { while (!entryPoints_.empty()) { std::string hostname = entryPoints_.front().first; if (util::isNumericHost(hostname)) { ++numSuccess_; std::pair p(hostname, entryPoints_.front().second); addPingTask(p); } else { std::vector res; int rv = resolveHostname(res, hostname); if (rv == 0) { e_->addCommand(std::unique_ptr(this)); return false; } else { if (rv == 1) { ++numSuccess_; std::pair p(res.front(), entryPoints_.front().second); addPingTask(p); } asyncNameResolverMan_->reset(e_, this); } } entryPoints_.pop_front(); } } else #endif // ENABLE_ASYNC_DNS { NameResolver res; res.setSocktype(SOCK_DGRAM); if (e_->getOption()->getAsBool(PREF_DISABLE_IPV6)) { res.setFamily(AF_INET); } while (!entryPoints_.empty()) { std::string hostname = entryPoints_.front().first; try { std::vector addrs; res.resolve(addrs, hostname); ++numSuccess_; std::pair p(addrs.front(), entryPoints_.front().second); addPingTask(p); } catch (RecoverableException& e) { A2_LOG_ERROR_EX(EX_EXCEPTION_CAUGHT, e); } entryPoints_.pop_front(); } } if (bootstrapEnabled_ && numSuccess_) { taskQueue_->addPeriodicTask1( taskFactory_->createNodeLookupTask(localNode_->getID())); taskQueue_->addPeriodicTask1(taskFactory_->createBucketRefreshTask()); } } catch (RecoverableException& e) { A2_LOG_ERROR_EX(EX_EXCEPTION_CAUGHT, e); } return true; } void DHTEntryPointNameResolveCommand::addPingTask( const std::pair& addr) { auto entryNode = std::make_shared(); entryNode->setIPAddress(addr.first); entryNode->setPort(addr.second); taskQueue_->addPeriodicTask1(taskFactory_->createPingTask(entryNode, 10)); } #ifdef ENABLE_ASYNC_DNS int DHTEntryPointNameResolveCommand::resolveHostname( std::vector& res, const std::string& hostname) { if (!asyncNameResolverMan_->started()) { asyncNameResolverMan_->startAsync(hostname, e_, this); } switch (asyncNameResolverMan_->getStatus()) { case -1: A2_LOG_INFO(fmt(MSG_NAME_RESOLUTION_FAILED, getCuid(), hostname.c_str(), asyncNameResolverMan_->getLastError().c_str())); return -1; case 0: return 0; case 1: asyncNameResolverMan_->getResolvedAddress(res); if (res.empty()) { A2_LOG_INFO(fmt(MSG_NAME_RESOLUTION_FAILED, getCuid(), hostname.c_str(), "No address returned")); return -1; } else { A2_LOG_INFO(fmt(MSG_NAME_RESOLUTION_COMPLETE, getCuid(), hostname.c_str(), res.front().c_str())); return 1; } } // Unreachable return 0; } #endif // ENABLE_ASYNC_DNS void DHTEntryPointNameResolveCommand::setBootstrapEnabled(bool f) { bootstrapEnabled_ = f; } void DHTEntryPointNameResolveCommand::setTaskQueue(DHTTaskQueue* taskQueue) { taskQueue_ = taskQueue; } void DHTEntryPointNameResolveCommand::setTaskFactory( DHTTaskFactory* taskFactory) { taskFactory_ = taskFactory; } void DHTEntryPointNameResolveCommand::setRoutingTable( DHTRoutingTable* routingTable) { routingTable_ = routingTable; } void DHTEntryPointNameResolveCommand::setLocalNode( const std::shared_ptr& localNode) { localNode_ = localNode; } } // namespace aria2