/* */ #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(std::begin(s->getRemainingUris()), std::end(s->getRemainingUris())); 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::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()) { // TODO this may be dead code 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(std::begin(destFiles), std::end(destFiles), 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(std::begin(destFiles), std::end(destFiles), e, EntryCmp()); if (d == std::end(destFiles) || (*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