/* */ #include "BtDependency.h" #include "RequestGroup.h" #include "Option.h" #include "LogFactory.h" #include "Logger.h" #include "DownloadContext.h" #include "RecoverableException.h" #include "message.h" #include "prefs.h" #include "util.h" #include "PieceStorage.h" #include "DiskAdaptor.h" #include "File.h" #include "bittorrent_helper.h" #include "DlAbortEx.h" #include "fmt.h" #include "FileEntry.h" #include "SimpleRandomizer.h" namespace aria2 { BtDependency::BtDependency (RequestGroup* dependant, const std::shared_ptr& dependee) : dependant_(dependant), dependee_(dependee) {} BtDependency::~BtDependency() {} namespace { void copyValues(const std::shared_ptr& d, const std::shared_ptr& s) { d->setRequested(true); d->setPath(s->getPath()); d->addUris(s->getRemainingUris().begin(), s->getRemainingUris().end()); d->setMaxConnectionPerServer(s->getMaxConnectionPerServer()); d->setUniqueProtocol(s->isUniqueProtocol()); } } // namespace namespace { struct EntryCmp { bool operator() (const std::shared_ptr& lhs, const std::shared_ptr& rhs) const { return lhs->getOriginalName() < rhs->getOriginalName(); } }; } // namespace bool BtDependency::resolve() { if(!dependee_) { return true; } if(dependee_->getNumCommand() == 0 && dependee_->downloadFinished()) { std::shared_ptr dependee = dependee_; // cut reference here dependee_.reset(); auto context = std::make_shared(); try { std::shared_ptr diskAdaptor = dependee->getPieceStorage()->getDiskAdaptor(); diskAdaptor->openExistingFile(); std::string content = util::toString(diskAdaptor); if(dependee->getDownloadContext()->hasAttribute(CTX_ATTR_BT)) { auto attrs = bittorrent::getTorrentAttrs(dependee->getDownloadContext()); bittorrent::loadFromMemory (bittorrent::metadata2Torrent(content, attrs), context, dependant_->getOption(), "default"); // We don't call bittorrent::adjustAnnounceUri() because it // has already been called with attrs. } else { bittorrent::loadFromMemory (content, context, dependant_->getOption(), File(dependee->getFirstFilePath()).getBasename()); bittorrent::adjustAnnounceUri(bittorrent::getTorrentAttrs(context), dependant_->getOption()); } const std::vector >& fileEntries = context->getFileEntries(); for (auto &fe : fileEntries) { auto &uri = fe->getRemainingUris(); std::random_shuffle(std::begin(uri), std::end(uri), *SimpleRandomizer::getInstance()); } const std::vector >& dependantFileEntries = dependant_->getDownloadContext()->getFileEntries(); // If dependant's FileEntry::getOriginalName() is empty, we // assume that torrent is single file. In Metalink3, this is // always assumed. if(fileEntries.size() == 1 && dependantFileEntries.size() == 1 && dependantFileEntries[0]->getOriginalName().empty()) { copyValues(fileEntries[0], dependantFileEntries[0]); } else { std::vector > destFiles; destFiles.reserve(fileEntries.size()); for(auto & e : fileEntries) { e->setRequested(false); destFiles.push_back(e); } std::sort(destFiles.begin(), destFiles.end(), EntryCmp()); // Copy file path in dependant_'s FileEntries to newly created // context's FileEntries to endorse the path structure of // dependant_. URIs and singleHostMultiConnection are also copied. for(const auto& e: dependantFileEntries){ const auto d = std::lower_bound(destFiles.begin(), destFiles.end(), e, EntryCmp()); if(d == destFiles.end() || (*d)->getOriginalName() != e->getOriginalName()) { throw DL_ABORT_EX (fmt("No entry %s in torrent file", e->getOriginalName().c_str())); } else { copyValues(*d, e); } } } } catch(RecoverableException& e) { A2_LOG_ERROR_EX(EX_EXCEPTION_CAUGHT, e); A2_LOG_INFO(fmt("BtDependency for GID#%s failed. Go without Bt.", GroupId::toHex(dependant_->getGID()).c_str())); return true; } A2_LOG_INFO(fmt("Dependency resolved for GID#%s", GroupId::toHex(dependant_->getGID()).c_str())); dependant_->setDownloadContext(context); return true; } else if(dependee_->getNumCommand() == 0) { // dependee_'s download failed. // cut reference here dependee_.reset(); A2_LOG_INFO(fmt("BtDependency for GID#%s failed. Go without Bt.", GroupId::toHex(dependant_->getGID()).c_str())); return true; } else { return false; } } } // namespace aria2