/* */ #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" 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(); std::shared_ptr context(new DownloadContext()); 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(); 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(std::vector >::const_iterator i = fileEntries.begin(), eoi = fileEntries.end(); i != eoi; ++i) { (*i)->setRequested(false); destFiles.push_back(*i); } 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(std::vector >::const_iterator s = dependantFileEntries.begin(), eoi = dependantFileEntries.end(); s != eoi; ++s){ std::vector >::const_iterator d = std::lower_bound(destFiles.begin(), destFiles.end(), *s, EntryCmp()); if(d == destFiles.end() || (*d)->getOriginalName() != (*s)->getOriginalName()) { throw DL_ABORT_EX (fmt("No entry %s in torrent file", (*s)->getOriginalName().c_str())); } else { copyValues(*d, *s); } } } } 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