/* */ #include "DHTEntryPointNameResolveCommand.h" #include "DownloadEngine.h" #ifdef ENABLE_ASYNC_DNS #include "AsyncNameResolver.h" #endif // ENABLE_ASYNC_DNS #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 "StringFormat.h" #include "FileEntry.h" #include "ServerStatMan.h" #include "FileAllocationEntry.h" #include "CheckIntegrityEntry.h" namespace aria2 { DHTEntryPointNameResolveCommand::DHTEntryPointNameResolveCommand (cuid_t cuid, DownloadEngine* e, const std::vector >& entryPoints): Command(cuid), _e(e), _entryPoints(entryPoints.begin(), entryPoints.end()), _bootstrapEnabled(false) {} DHTEntryPointNameResolveCommand::~DHTEntryPointNameResolveCommand() { #ifdef ENABLE_ASYNC_DNS disableNameResolverCheck(_resolver); #endif // ENABLE_ASYNC_DNS } bool DHTEntryPointNameResolveCommand::execute() { if(_e->getRequestGroupMan()->downloadFinished() || _e->isHaltRequested()) { return true; } #ifdef ENABLE_ASYNC_DNS if(_resolver.isNull()) { _resolver.reset(new AsyncNameResolver()); } #endif // ENABLE_ASYNC_DNS try { #ifdef ENABLE_ASYNC_DNS if(_e->getOption()->getAsBool(PREF_ASYNC_DNS)) { while(!_entryPoints.empty()) { std::string hostname = _entryPoints.front().first; try { if(resolveHostname(hostname, _resolver)) { hostname = _resolver->getResolvedAddresses().front(); std::pair p(hostname, _entryPoints.front().second); _resolvedEntryPoints.push_back(p); addPingTask(p); } else { _e->addCommand(this); return false; } } catch(RecoverableException& e) { getLogger()->error(EX_EXCEPTION_CAUGHT, e); } _resolver->reset(); _entryPoints.pop_front(); } } else #endif // ENABLE_ASYNC_DNS { NameResolver res; res.setSocktype(SOCK_DGRAM); while(!_entryPoints.empty()) { std::string hostname = _entryPoints.front().first; try { std::vector addrs; res.resolve(addrs, hostname); std::pair p(addrs.front(), _entryPoints.front().second); _resolvedEntryPoints.push_back(p); addPingTask(p); } catch(RecoverableException& e) { getLogger()->error(EX_EXCEPTION_CAUGHT, e); } _entryPoints.pop_front(); } } if(_bootstrapEnabled && _resolvedEntryPoints.size()) { _taskQueue->addPeriodicTask1(_taskFactory->createNodeLookupTask (_localNode->getID())); _taskQueue->addPeriodicTask1(_taskFactory->createBucketRefreshTask()); } } catch(RecoverableException& e) { getLogger()->error(EX_EXCEPTION_CAUGHT, e); } return true; } void DHTEntryPointNameResolveCommand::addPingTask (const std::pair& addr) { SharedHandle entryNode(new DHTNode()); entryNode->setIPAddress(addr.first); entryNode->setPort(addr.second); _taskQueue->addPeriodicTask1(_taskFactory->createPingTask(entryNode, 10)); } #ifdef ENABLE_ASYNC_DNS bool DHTEntryPointNameResolveCommand::resolveHostname (const std::string& hostname, const SharedHandle& resolver) { switch(resolver->getStatus()) { case AsyncNameResolver::STATUS_READY: if(getLogger()->info()) { getLogger()->info(MSG_RESOLVING_HOSTNAME, util::itos(getCuid()).c_str(), hostname.c_str()); } resolver->resolve(hostname); setNameResolverCheck(resolver); return false; case AsyncNameResolver::STATUS_SUCCESS: if(getLogger()->info()) { getLogger()->info(MSG_NAME_RESOLUTION_COMPLETE, util::itos(getCuid()).c_str(), resolver->getHostname().c_str(), resolver->getResolvedAddresses().front().c_str()); } return true; break; case AsyncNameResolver::STATUS_ERROR: throw DL_ABORT_EX (StringFormat(MSG_NAME_RESOLUTION_FAILED, util::itos(getCuid()).c_str(), hostname.c_str(), resolver->getError().c_str()).str()); default: return false; } } void DHTEntryPointNameResolveCommand::setNameResolverCheck (const SharedHandle& resolver) { _e->addNameResolverCheck(resolver, this); } void DHTEntryPointNameResolveCommand::disableNameResolverCheck (const SharedHandle& resolver) { _e->deleteNameResolverCheck(resolver, this); } #endif // ENABLE_ASYNC_DNS void DHTEntryPointNameResolveCommand::setBootstrapEnabled(bool f) { _bootstrapEnabled = f; } void DHTEntryPointNameResolveCommand::setTaskQueue (const SharedHandle& taskQueue) { _taskQueue = taskQueue; } void DHTEntryPointNameResolveCommand::setTaskFactory (const SharedHandle& taskFactory) { _taskFactory = taskFactory; } void DHTEntryPointNameResolveCommand::setRoutingTable (const SharedHandle& routingTable) { _routingTable = routingTable; } void DHTEntryPointNameResolveCommand::setLocalNode (const SharedHandle& localNode) { _localNode = localNode; } } // namespace aria2