/* */ #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 "StringFormat.h" namespace aria2 { BtDependency::BtDependency(const WeakHandle& dependant, const SharedHandle& dependee): _dependant(dependant), _dependee(dependee), _logger(LogFactory::getInstance()) {} BtDependency::~BtDependency() {} static void copyValues(const SharedHandle& d, const SharedHandle& s) { d->setRequested(true); d->setPath(s->getPath()); d->addUris(s->getRemainingUris().begin(), s->getRemainingUris().end()); if(!s->isSingleHostMultiConnectionEnabled()) { d->disableSingleHostMultiConnection(); } } bool BtDependency::resolve() { if(_dependee->getNumCommand() == 0 && _dependee->downloadFinished()) { SharedHandle dependee = _dependee; // cut reference here _dependee.reset(); SharedHandle context(new DownloadContext()); context->setDir(_dependant->getDownloadContext()->getDir()); try { SharedHandle diskAdaptor = dependee->getPieceStorage()->getDiskAdaptor(); diskAdaptor->openExistingFile(); std::string content = util::toString(diskAdaptor); if(dependee->getDownloadContext()->hasAttribute(bittorrent::BITTORRENT)) { const BDE& attrs = dependee->getDownloadContext()->getAttribute(bittorrent::BITTORRENT); bittorrent::loadFromMemory (bittorrent::metadata2Torrent(content, attrs), context, "default"); } else { bittorrent::loadFromMemory (content, context, File(dependee->getFirstFilePath()).getBasename()); } 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::for_each(fileEntries.begin(), fileEntries.end(), std::bind2nd(mem_fun_sh(&FileEntry::setRequested),false)); // 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. std::vector >::const_iterator ctxFilesEnd = fileEntries.end(); for(std::vector >::const_iterator s = dependantFileEntries.begin(), eoi = dependantFileEntries.end(); s != eoi; ++s){ std::vector >::const_iterator d = fileEntries.begin(); for(; d != ctxFilesEnd; ++d) { if((*d)->getOriginalName() == (*s)->getOriginalName()) { break; } } if(d == ctxFilesEnd) { throw DL_ABORT_EX (StringFormat("No entry %s in torrent file", (*s)->getOriginalName().c_str()).str()); } copyValues(*d, *s); } } } catch(RecoverableException& e) { _logger->error(EX_EXCEPTION_CAUGHT, e); if(_logger->info()) { _logger->info("BtDependency for GID#%s failed. Go without Bt.", util::itos(_dependant->getGID()).c_str()); } return true; } if(_logger->info()) { _logger->info("Dependency resolved for GID#%s", util::itos(_dependant->getGID()).c_str()); } _dependant->setDownloadContext(context); return true; } else if(_dependee->getNumCommand() == 0) { // _dependee's download failed. // cut reference here _dependee.reset(); if(_logger->info()) { _logger->info("BtDependency for GID#%s failed. Go without Bt.", util::itos(_dependant->getGID()).c_str()); } return true; } else { return false; } } } // namespace aria2