BtDependency.cc 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. /* <!-- copyright */
  2. /*
  3. * aria2 - The high speed download utility
  4. *
  5. * Copyright (C) 2006 Tatsuhiro Tsujikawa
  6. *
  7. * This program is free software; you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License as published by
  9. * the Free Software Foundation; either version 2 of the License, or
  10. * (at your option) any later version.
  11. *
  12. * This program is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License
  18. * along with this program; if not, write to the Free Software
  19. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  20. *
  21. * In addition, as a special exception, the copyright holders give
  22. * permission to link the code of portions of this program with the
  23. * OpenSSL library under certain conditions as described in each
  24. * individual source file, and distribute linked combinations
  25. * including the two.
  26. * You must obey the GNU General Public License in all respects
  27. * for all of the code used other than OpenSSL. If you modify
  28. * file(s) with this exception, you may extend this exception to your
  29. * version of the file(s), but you are not obligated to do so. If you
  30. * do not wish to do so, delete this exception statement from your
  31. * version. If you delete this exception statement from all source
  32. * files in the program, then also delete it here.
  33. */
  34. /* copyright --> */
  35. #include "BtDependency.h"
  36. #include "RequestGroup.h"
  37. #include "Option.h"
  38. #include "LogFactory.h"
  39. #include "Logger.h"
  40. #include "DownloadContext.h"
  41. #include "RecoverableException.h"
  42. #include "message.h"
  43. #include "prefs.h"
  44. #include "util.h"
  45. #include "PieceStorage.h"
  46. #include "DiskAdaptor.h"
  47. #include "File.h"
  48. #include "bittorrent_helper.h"
  49. #include "DlAbortEx.h"
  50. #include "fmt.h"
  51. #include "FileEntry.h"
  52. #include "SimpleRandomizer.h"
  53. namespace aria2 {
  54. BtDependency::BtDependency(RequestGroup* dependant,
  55. const std::shared_ptr<RequestGroup>& dependee)
  56. : dependant_(dependant), dependee_(dependee)
  57. {
  58. }
  59. BtDependency::~BtDependency() = default;
  60. namespace {
  61. void copyValues(const std::shared_ptr<FileEntry>& d,
  62. const std::shared_ptr<FileEntry>& s)
  63. {
  64. d->setRequested(true);
  65. d->setPath(s->getPath());
  66. d->addUris(std::begin(s->getRemainingUris()),
  67. std::end(s->getRemainingUris()));
  68. d->setMaxConnectionPerServer(s->getMaxConnectionPerServer());
  69. d->setUniqueProtocol(s->isUniqueProtocol());
  70. }
  71. } // namespace
  72. namespace {
  73. struct EntryCmp {
  74. bool operator()(const std::shared_ptr<FileEntry>& lhs,
  75. const std::shared_ptr<FileEntry>& rhs) const
  76. {
  77. return lhs->getOriginalName() < rhs->getOriginalName();
  78. }
  79. };
  80. } // namespace
  81. bool BtDependency::resolve()
  82. {
  83. if (!dependee_) {
  84. return true;
  85. }
  86. if (dependee_->getNumCommand() == 0 && dependee_->downloadFinished()) {
  87. std::shared_ptr<RequestGroup> dependee = dependee_;
  88. // cut reference here
  89. dependee_.reset();
  90. auto context = std::make_shared<DownloadContext>();
  91. try {
  92. std::shared_ptr<DiskAdaptor> diskAdaptor =
  93. dependee->getPieceStorage()->getDiskAdaptor();
  94. diskAdaptor->openExistingFile();
  95. std::string content = util::toString(diskAdaptor);
  96. if (dependee->getDownloadContext()->hasAttribute(CTX_ATTR_BT)) {
  97. auto attrs =
  98. bittorrent::getTorrentAttrs(dependee->getDownloadContext());
  99. bittorrent::loadFromMemory(bittorrent::metadata2Torrent(content, attrs),
  100. context, dependant_->getOption(), "default");
  101. // We don't call bittorrent::adjustAnnounceUri() because it
  102. // has already been called with attrs.
  103. }
  104. else {
  105. bittorrent::loadFromMemory(
  106. content, context, dependant_->getOption(),
  107. File(dependee->getFirstFilePath()).getBasename());
  108. bittorrent::adjustAnnounceUri(bittorrent::getTorrentAttrs(context),
  109. dependant_->getOption());
  110. }
  111. const std::vector<std::shared_ptr<FileEntry>>& fileEntries =
  112. context->getFileEntries();
  113. for (auto& fe : fileEntries) {
  114. auto& uri = fe->getRemainingUris();
  115. std::shuffle(std::begin(uri), std::end(uri),
  116. *SimpleRandomizer::getInstance());
  117. }
  118. const std::vector<std::shared_ptr<FileEntry>>& dependantFileEntries =
  119. dependant_->getDownloadContext()->getFileEntries();
  120. // If dependant's FileEntry::getOriginalName() is empty, we
  121. // assume that torrent is single file. In Metalink3, this is
  122. // always assumed.
  123. if (fileEntries.size() == 1 && dependantFileEntries.size() == 1 &&
  124. dependantFileEntries[0]->getOriginalName().empty()) {
  125. // TODO this may be dead code
  126. copyValues(fileEntries[0], dependantFileEntries[0]);
  127. }
  128. else {
  129. std::vector<std::shared_ptr<FileEntry>> destFiles;
  130. destFiles.reserve(fileEntries.size());
  131. for (auto& e : fileEntries) {
  132. e->setRequested(false);
  133. destFiles.push_back(e);
  134. }
  135. std::sort(std::begin(destFiles), std::end(destFiles), EntryCmp());
  136. // Copy file path in dependant_'s FileEntries to newly created
  137. // context's FileEntries to endorse the path structure of
  138. // dependant_. URIs and singleHostMultiConnection are also copied.
  139. for (const auto& e : dependantFileEntries) {
  140. const auto d = std::lower_bound(std::begin(destFiles),
  141. std::end(destFiles), e, EntryCmp());
  142. if (d == std::end(destFiles) ||
  143. (*d)->getOriginalName() != e->getOriginalName()) {
  144. throw DL_ABORT_EX(fmt("No entry %s in torrent file",
  145. e->getOriginalName().c_str()));
  146. }
  147. else {
  148. copyValues(*d, e);
  149. }
  150. }
  151. }
  152. }
  153. catch (RecoverableException& e) {
  154. A2_LOG_ERROR_EX(EX_EXCEPTION_CAUGHT, e);
  155. A2_LOG_INFO(fmt("BtDependency for GID#%s failed. Go without Bt.",
  156. GroupId::toHex(dependant_->getGID()).c_str()));
  157. return true;
  158. }
  159. A2_LOG_INFO(fmt("Dependency resolved for GID#%s",
  160. GroupId::toHex(dependant_->getGID()).c_str()));
  161. dependant_->setDownloadContext(context);
  162. return true;
  163. }
  164. else if (dependee_->getNumCommand() == 0) {
  165. // dependee_'s download failed.
  166. // cut reference here
  167. dependee_.reset();
  168. A2_LOG_INFO(fmt("BtDependency for GID#%s failed. Go without Bt.",
  169. GroupId::toHex(dependant_->getGID()).c_str()));
  170. return true;
  171. }
  172. else {
  173. return false;
  174. }
  175. }
  176. } // namespace aria2