/* */ #include "HttpListenCommand.h" #include "DownloadEngine.h" #include "RecoverableException.h" #include "message.h" #include "Logger.h" #include "LogFactory.h" #include "SocketCore.h" #include "HttpServerCommand.h" #include "CUIDCounter.h" #include "RequestGroupMan.h" #include "prefs.h" #include "Option.h" #include "util.h" #include "A2STR.h" #include "fmt.h" namespace aria2 { HttpListenCommand::HttpListenCommand(cuid_t cuid, DownloadEngine* e, int family) : Command(cuid), e_(e), family_(family) {} HttpListenCommand::~HttpListenCommand() { if(serverSocket_) { e_->deleteSocketForReadCheck(serverSocket_, this); } } bool HttpListenCommand::execute() { if(e_->getRequestGroupMan()->downloadFinished() || e_->isHaltRequested()) { return true; } try { if(serverSocket_->isReadable(0)) { SharedHandle socket(serverSocket_->acceptConnection()); socket->setNonBlockingMode(); std::pair peerInfo; socket->getPeerInfo(peerInfo); A2_LOG_INFO(fmt("XML-RPC: Accepted the connection from %s:%u.", peerInfo.first.c_str(), peerInfo.second)); HttpServerCommand* c = new HttpServerCommand(e_->newCUID(), e_, socket); e_->setNoWait(true); e_->addCommand(c); } } catch(RecoverableException& e) { A2_LOG_DEBUG_EX(fmt(MSG_ACCEPT_FAILURE, getCuid()), e); } e_->addCommand(this); return false; } bool HttpListenCommand::bindPort(uint16_t port) { if(serverSocket_) { e_->deleteSocketForReadCheck(serverSocket_, this); } serverSocket_.reset(new SocketCore()); A2_LOG_INFO(fmt("CUID#%lld - Setting up HttpListenCommand for IPv%d", getCuid(), family_ == AF_INET?4:6)); try { int flags = 0; if(e_->getOption()->getAsBool(PREF_XML_RPC_LISTEN_ALL)) { flags = AI_PASSIVE; } serverSocket_->bind(A2STR::NIL, port, family_, flags); serverSocket_->beginListen(); serverSocket_->setNonBlockingMode(); A2_LOG_INFO(fmt(MSG_LISTENING_PORT, getCuid(), port)); e_->addSocketForReadCheck(serverSocket_, this); return true; } catch(RecoverableException& e) { A2_LOG_ERROR(fmt("Failed to setup XML-RPC server for IPv%d", family_ == AF_INET?4:6)); A2_LOG_ERROR_EX(fmt(MSG_BIND_FAILURE, getCuid(), port), e); if(serverSocket_) { e_->deleteSocketForReadCheck(serverSocket_, this); } serverSocket_->closeConnection(); } return false; } } // namespace aria2