/* */ #include "InitiateConnectionCommand.h" #include "Request.h" #include "DownloadEngine.h" #include "Option.h" #include "Logger.h" #include "LogFactory.h" #include "message.h" #include "prefs.h" #include "NameResolver.h" #include "SocketCore.h" #include "FileEntry.h" #include "RequestGroup.h" #include "Segment.h" #include "a2functional.h" #include "InitiateConnectionCommandFactory.h" #include "util.h" #include "RecoverableException.h" #include "fmt.h" #include "SocketRecvBuffer.h" #include "BackupIPv4ConnectCommand.h" #include "ConnectCommand.h" namespace aria2 { InitiateConnectionCommand::InitiateConnectionCommand (cuid_t cuid, const std::shared_ptr& req, const std::shared_ptr& fileEntry, RequestGroup* requestGroup, DownloadEngine* e) : AbstractCommand(cuid, req, fileEntry, requestGroup, e) { setTimeout(getOption()->getAsInt(PREF_DNS_TIMEOUT)); // give a chance to be executed in the next loop in DownloadEngine setStatus(Command::STATUS_ONESHOT_REALTIME); disableReadCheckSocket(); disableWriteCheckSocket(); } InitiateConnectionCommand::~InitiateConnectionCommand() {} bool InitiateConnectionCommand::executeInternal() { std::string hostname; uint16_t port; std::shared_ptr proxyRequest = createProxyRequest(); if(!proxyRequest) { hostname = getRequest()->getHost(); port = getRequest()->getPort(); } else { hostname = proxyRequest->getHost(); port = proxyRequest->getPort(); } std::vector addrs; std::string ipaddr = resolveHostname(addrs, hostname, port); if(ipaddr.empty()) { addCommandSelf(); return false; } try { getDownloadEngine()->addCommand(createNextCommand(hostname, ipaddr, port, addrs, proxyRequest)); return true; } catch(RecoverableException& ex) { // Catch exception and retry another address. // See also AbstractCommand::checkIfConnectionEstablished // TODO ipaddr might not be used if pooled socket was found. getDownloadEngine()->markBadIPAddress(hostname, ipaddr, port); if(!getDownloadEngine()->findCachedIPAddress(hostname, port).empty()) { A2_LOG_INFO_EX(EX_EXCEPTION_CAUGHT, ex); A2_LOG_INFO(fmt(MSG_CONNECT_FAILED_AND_RETRY, getCuid(), ipaddr.c_str(), port)); auto command = InitiateConnectionCommandFactory::createInitiateConnectionCommand (getCuid(), getRequest(), getFileEntry(), getRequestGroup(), getDownloadEngine()); getDownloadEngine()->setNoWait(true); getDownloadEngine()->addCommand(std::move(command)); return true; } getDownloadEngine()->removeCachedIPAddress(hostname, port); throw; } } void InitiateConnectionCommand::setConnectedAddrInfo (const std::shared_ptr& req, const std::string& hostname, const std::shared_ptr& socket) { std::pair peerAddr; socket->getPeerInfo(peerAddr); req->setConnectedAddrInfo(hostname, peerAddr.first, peerAddr.second); } std::shared_ptr InitiateConnectionCommand::createBackupIPv4ConnectCommand (const std::string& hostname, const std::string& ipaddr, uint16_t port, Command* mainCommand) { // Prepare IPv4 backup connection attemp in "Happy Eyeballs" // fashion. std::shared_ptr info; char buf[sizeof(in6_addr)]; if(inetPton(AF_INET6, ipaddr.c_str(), &buf) == -1) { return info; } A2_LOG_INFO("Searching IPv4 address for backup connection attempt"); std::vector addrs; getDownloadEngine()->findAllCachedIPAddresses(std::back_inserter(addrs), hostname, port); for(std::vector::const_iterator i = addrs.begin(), eoi = addrs.end(); i != eoi; ++i) { if(inetPton(AF_INET, (*i).c_str(), &buf) == 0) { info.reset(new BackupConnectInfo()); auto command = make_unique (getDownloadEngine()->newCUID(), *i, port, info, mainCommand, getRequestGroup(), getDownloadEngine()); A2_LOG_INFO(fmt("Issue backup connection command CUID#%" PRId64 ", addr=%s", command->getCuid(), (*i).c_str())); getDownloadEngine()->addCommand(std::move(command)); return info; } } return info; } void InitiateConnectionCommand::setupBackupConnection (const std::string& hostname, const std::string& addr, uint16_t port, ConnectCommand* c) { std::shared_ptr backupConnectInfo = createBackupIPv4ConnectCommand(hostname, addr, port, c); if(backupConnectInfo) { c->setBackupConnectInfo(backupConnectInfo); } } } // namespace aria2