/* */ #include "CreateRequestCommand.h" #include "InitiateConnectionCommandFactory.h" #include "RequestGroup.h" #include "Segment.h" #include "DownloadContext.h" #include "DlAbortEx.h" #include "DownloadEngine.h" #include "SocketCore.h" #include "SegmentMan.h" #include "prefs.h" #include "Option.h" #include "RequestGroupMan.h" #include "FileEntry.h" #include "SocketRecvBuffer.h" #include "LogFactory.h" #include "wallclock.h" #include "DownloadFailureException.h" namespace aria2 { CreateRequestCommand::CreateRequestCommand(cuid_t cuid, RequestGroup* requestGroup, DownloadEngine* e) : AbstractCommand(cuid, std::shared_ptr(), std::shared_ptr(), requestGroup, e, std::shared_ptr(), std::shared_ptr(), false) { setStatus(Command::STATUS_ONESHOT_REALTIME); disableReadCheckSocket(); disableWriteCheckSocket(); } bool CreateRequestCommand::executeInternal() { if (getSegments().empty()) { setFileEntry(getDownloadContext()->findFileEntryByOffset(0)); } else { // We assume all segments belongs to same file. setFileEntry(getDownloadContext()->findFileEntryByOffset( getSegments().front()->getPositionToWrite())); } std::vector> usedHosts; if (getOption()->getAsBool(PREF_SELECT_LEAST_USED_HOST)) { getDownloadEngine()->getRequestGroupMan()->getUsedHosts(usedHosts); } setRequest( getFileEntry()->getRequest(getRequestGroup()->getURISelector().get(), getOption()->getAsBool(PREF_REUSE_URI), usedHosts, getOption()->get(PREF_REFERER), // Don't use HEAD request when file // size is known. // Use HEAD for dry-run mode. (getFileEntry()->getLength() == 0 && getOption()->getAsBool(PREF_USE_HEAD)) || getOption()->getAsBool(PREF_DRY_RUN) ? Request::METHOD_HEAD : Request::METHOD_GET)); if (!getRequest()) { if (getSegmentMan()) { getSegmentMan()->ignoreSegmentFor(getFileEntry()); } if (getOption()->getAsBool(PREF_DRY_RUN)) { // For dry run mode, just throwing DlAbortEx makes infinite // loop, since we don't have SegmentMan, and we cannot tell all // hopes were abandoned. throw DOWNLOAD_FAILURE_EXCEPTION2("No URI available.", getRequestGroup()->getLastErrorCode()); } // In this case, the error might be already set in RequestGroup, // so use it here. throw DL_ABORT_EX2("No URI available.", getRequestGroup()->getLastErrorCode()); } else if (getRequest()->getWakeTime() > global::wallclock()) { A2_LOG_DEBUG("This request object is still sleeping."); getFileEntry()->poolRequest(getRequest()); // Reset request of this command. Without this, request is doubly // counted (1 for pooled and another one in this command) and // AbstractCommand::execute() will behave badly. resetRequest(); addCommandSelf(); return false; } getDownloadEngine()->setNoWait(true); getDownloadEngine()->addCommand( InitiateConnectionCommandFactory::createInitiateConnectionCommand( getCuid(), getRequest(), getFileEntry(), getRequestGroup(), getDownloadEngine())); return true; } } // namespace aria2