2010-02-27 Tatsuhiro Tsujikawa <t-tujikawa@users.sourceforge.net>

Supported name attribute of metalink::metaurl element and
	multi-file torrent with Metalink4.  Files with same metaurl are
	grouped and downloaded in one RequestGroup.
	* src/BtDependency.cc
	* src/FileEntry.h
	* src/Metalink2RequestGroup.cc
	* src/Metalink2RequestGroup.h
	* src/MetalinkEntry.cc
	* src/MetalinkEntry.h
	* src/MetalinkHelper.cc
	* src/MetalinkHelper.h
	* src/MetalinkMetaurl.cc
	* src/MetalinkMetaurl.h
	* src/MetalinkParserController.cc
	* src/RequestGroup.cc
	* src/RequestGroup.h
	* src/bittorrent_helper.cc
	* test/BittorrentHelperTest.cc
	* test/BtDependencyTest.cc
	* test/MetalinkHelperTest.cc
This commit is contained in:
Tatsuhiro Tsujikawa 2010-02-26 15:37:08 +00:00
parent 5032394c6a
commit 2a6775e80b
18 changed files with 500 additions and 157 deletions

View file

@ -1,3 +1,26 @@
2010-02-27 Tatsuhiro Tsujikawa <t-tujikawa@users.sourceforge.net>
Supported name attribute of metalink::metaurl element and
multi-file torrent with Metalink4. Files with same metaurl are
grouped and downloaded in one RequestGroup.
* src/BtDependency.cc
* src/FileEntry.h
* src/Metalink2RequestGroup.cc
* src/Metalink2RequestGroup.h
* src/MetalinkEntry.cc
* src/MetalinkEntry.h
* src/MetalinkHelper.cc
* src/MetalinkHelper.h
* src/MetalinkMetaurl.cc
* src/MetalinkMetaurl.h
* src/MetalinkParserController.cc
* src/RequestGroup.cc
* src/RequestGroup.h
* src/bittorrent_helper.cc
* test/BittorrentHelperTest.cc
* test/BtDependencyTest.cc
* test/MetalinkHelperTest.cc
2010-02-26 Tatsuhiro Tsujikawa <t-tujikawa@users.sourceforge.net> 2010-02-26 Tatsuhiro Tsujikawa <t-tujikawa@users.sourceforge.net>
Store name attribute of metalink:metaurl element in MetalinkMetaurl. Store name attribute of metalink:metaurl element in MetalinkMetaurl.

View file

@ -47,6 +47,7 @@
#include "File.h" #include "File.h"
#include "bittorrent_helper.h" #include "bittorrent_helper.h"
#include "DlAbortEx.h" #include "DlAbortEx.h"
#include "StringFormat.h"
namespace aria2 { namespace aria2 {
@ -58,6 +59,18 @@ BtDependency::BtDependency(const RequestGroupWeakHandle& dependant,
BtDependency::~BtDependency() {} BtDependency::~BtDependency() {}
static void copyValues(const SharedHandle<FileEntry>& d,
const SharedHandle<FileEntry>& s)
{
d->setRequested(true);
d->setPath(s->getPath());
d->addUris(s->getRemainingUris().begin(),
s->getRemainingUris().end());
if(!s->isSingleHostMultiConnectionEnabled()) {
d->disableSingleHostMultiConnection();
}
}
bool BtDependency::resolve() bool BtDependency::resolve()
{ {
if(_dependee->getNumCommand() == 0 && _dependee->downloadFinished()) { if(_dependee->getNumCommand() == 0 && _dependee->downloadFinished()) {
@ -80,23 +93,38 @@ bool BtDependency::resolve()
bittorrent::loadFromMemory bittorrent::loadFromMemory
(content, context, File(dependee->getFirstFilePath()).getBasename()); (content, context, File(dependee->getFirstFilePath()).getBasename());
} }
if(context->getFileEntries().size() != const std::vector<SharedHandle<FileEntry> >& fileEntries =
_dependant->getDownloadContext()->getFileEntries().size()) { context->getFileEntries();
throw DL_ABORT_EX("The number of file in torrent doesn't match to" const std::vector<SharedHandle<FileEntry> >& dependantFileEntries =
" the dependent."); _dependant->getDownloadContext()->getFileEntries();
} // If dependant's FileEntry::getOriginalName() is empty, we
// Copy file path in _dependant's FileEntries to newly created // assume that torrent is single file. In Metalink3, this is
// context's FileEntries to endorse the path structure of // always assumed.
// _dependant. URIs and singleHostMultiConnection are also copied. if(fileEntries.size() == 1 && dependantFileEntries.size() == 1 &&
for(std::vector<SharedHandle<FileEntry> >::const_iterator s = dependantFileEntries[0]->getOriginalName().empty()) {
_dependant->getDownloadContext()->getFileEntries().begin(), copyValues(fileEntries[0], dependantFileEntries[0]);
d = context->getFileEntries().begin(); } else {
d != context->getFileEntries().end(); ++s, ++d) { std::for_each(fileEntries.begin(), fileEntries.end(),
(*d)->setPath((*s)->getPath()); std::bind2nd(mem_fun_sh(&FileEntry::setRequested),false));
(*d)->addUris((*s)->getRemainingUris().begin(), // Copy file path in _dependant's FileEntries to newly created
(*s)->getRemainingUris().end()); // context's FileEntries to endorse the path structure of
if(!(*s)->isSingleHostMultiConnectionEnabled()) { // _dependant. URIs and singleHostMultiConnection are also copied.
(*d)->disableSingleHostMultiConnection(); for(std::vector<SharedHandle<FileEntry> >::const_iterator s =
dependantFileEntries.begin(); s != dependantFileEntries.end();
++s){
std::vector<SharedHandle<FileEntry> >::const_iterator d =
context->getFileEntries().begin();
for(; d != context->getFileEntries().end(); ++d) {
if((*d)->getOriginalName() == (*s)->getOriginalName()) {
break;
}
}
if(d == context->getFileEntries().end()) {
throw DL_ABORT_EX
(StringFormat("No entry %s in torrent file",
(*s)->getOriginalName().c_str()).str());
}
copyValues(*d, *s);
} }
} }
} catch(RecoverableException& e) { } catch(RecoverableException& e) {

View file

@ -70,6 +70,7 @@ private:
// available. // available.
std::deque<URIResult> _uriResults; std::deque<URIResult> _uriResults;
bool _singleHostMultiConnection; bool _singleHostMultiConnection;
std::string _originalName;
Logger* _logger; Logger* _logger;
void storePool(const SharedHandle<Request>& request); void storePool(const SharedHandle<Request>& request);
@ -227,6 +228,16 @@ public:
void reuseUri(size_t num); void reuseUri(size_t num);
void releaseRuntimeResource(); void releaseRuntimeResource();
void setOriginalName(const std::string& originalName)
{
_originalName = originalName;
}
const std::string& getOriginalName() const
{
return _originalName;
}
}; };
typedef SharedHandle<FileEntry> FileEntryHandle; typedef SharedHandle<FileEntry> FileEntryHandle;

View file

@ -136,57 +136,66 @@ void removeMetalinkContentTypes(const SharedHandle<RequestGroup>& group)
void void
Metalink2RequestGroup::createRequestGroup Metalink2RequestGroup::createRequestGroup
(std::deque<SharedHandle<RequestGroup> >& groups, (std::deque<SharedHandle<RequestGroup> >& groups,
std::deque<SharedHandle<MetalinkEntry> > entries, const std::deque<SharedHandle<MetalinkEntry> >& entries,
const SharedHandle<Option>& option) const SharedHandle<Option>& option)
{ {
if(entries.size() == 0) { if(entries.empty()) {
_logger->notice(EX_NO_RESULT_WITH_YOUR_PREFS); _logger->notice(EX_NO_RESULT_WITH_YOUR_PREFS);
return; return;
} }
std::deque<int32_t> selectIndexes = std::deque<int32_t> selectIndexes =
util::parseIntRange(option->get(PREF_SELECT_FILE)).flush(); util::parseIntRange(option->get(PREF_SELECT_FILE)).flush();
bool useIndex; std::sort(selectIndexes.begin(), selectIndexes.end());
if(selectIndexes.size()) { std::vector<SharedHandle<MetalinkEntry> > selectedEntries;
useIndex = true; selectedEntries.reserve(entries.size());
} else {
useIndex = false; std::deque<std::string> locations;
if(option->defined(PREF_METALINK_LOCATION)) {
util::split(option->get(PREF_METALINK_LOCATION),
std::back_inserter(locations), ",", true);
std::transform
(locations.begin(), locations.end(), locations.begin(), util::toLower);
} }
int32_t count = 0; std::string preferredProtocol;
for(std::deque<SharedHandle<MetalinkEntry> >::iterator itr = entries.begin(); itr != entries.end(); if(option->get(PREF_METALINK_PREFERRED_PROTOCOL) != V_NONE) {
++itr, ++count) { preferredProtocol = option->get(PREF_METALINK_PREFERRED_PROTOCOL);
SharedHandle<MetalinkEntry>& entry = *itr; }
if(option->defined(PREF_METALINK_LOCATION)) { {
std::deque<std::string> locations; int32_t count = 1;
util::split(option->get(PREF_METALINK_LOCATION), for(std::deque<SharedHandle<MetalinkEntry> >::const_iterator i =
std::back_inserter(locations), ",", true); entries.begin(); i != entries.end(); ++i, ++count) {
std::transform (*i)->dropUnsupportedResource();
(locations.begin(), locations.end(), locations.begin(), util::toLower); if((*i)->resources.empty() && (*i)->metaurls.empty()) {
entry->setLocationPriority
(locations, -MetalinkResource::getLowestPriority());
}
if(option->get(PREF_METALINK_PREFERRED_PROTOCOL) != V_NONE) {
entry->setProtocolPriority
(option->get(PREF_METALINK_PREFERRED_PROTOCOL),
-MetalinkResource::getLowestPriority());
}
if(useIndex) {
if(std::find(selectIndexes.begin(), selectIndexes.end(), count+1) ==
selectIndexes.end()) {
continue; continue;
} }
(*i)->setLocationPriority
(locations, -MetalinkResource::getLowestPriority());
if(!preferredProtocol.empty()) {
(*i)->setProtocolPriority
(preferredProtocol, -MetalinkResource::getLowestPriority());
}
if(selectIndexes.empty() ||
std::binary_search(selectIndexes.begin(), selectIndexes.end(), count)){
selectedEntries.push_back(*i);
}
} }
entry->dropUnsupportedResource(); }
if(entry->resources.empty() && entry->metaurls.empty()) { std::for_each(entries.begin(), entries.end(),
continue; mem_fun_sh(&MetalinkEntry::reorderMetaurlsByPriority));
} std::vector<std::pair<std::string,
_logger->info(MSG_METALINK_QUEUEING, entry->getPath().c_str()); std::vector<SharedHandle<MetalinkEntry> > > > entryGroups;
MetalinkHelper::groupEntryByMetaurlName(entryGroups, selectedEntries);
for(std::vector<std::pair<std::string,
std::vector<SharedHandle<MetalinkEntry> > > >::const_iterator itr =
entryGroups.begin(); itr != entryGroups.end(); ++itr) {
const std::string& metaurl = (*itr).first;
const std::vector<SharedHandle<MetalinkEntry> >& mes = (*itr).second;
_logger->info("Processing metaurl group metaurl=%s", metaurl.c_str());
#ifdef ENABLE_BITTORRENT #ifdef ENABLE_BITTORRENT
SharedHandle<RequestGroup> torrentRg; SharedHandle<RequestGroup> torrentRg;
if(!entry->metaurls.empty()) { if(!metaurl.empty()) {
entry->reorderMetaurlsByPriority();
// there is torrent entry
std::deque<std::string> uris; std::deque<std::string> uris;
uris.push_back(entry->metaurls[0]->url); uris.push_back(metaurl);
{ {
std::deque<SharedHandle<RequestGroup> > result; std::deque<SharedHandle<RequestGroup> > result;
createRequestGroupForUri(result, option, uris, createRequestGroupForUri(result, option, uris,
@ -213,62 +222,94 @@ Metalink2RequestGroup::createRequestGroup
} }
} }
#endif // ENABLE_BITTORRENT #endif // ENABLE_BITTORRENT
entry->reorderResourcesByPriority();
std::deque<std::string> uris;
std::for_each(entry->resources.begin(), entry->resources.end(),
AccumulateNonP2PUrl(uris));
SharedHandle<RequestGroup> rg(new RequestGroup(option)); SharedHandle<RequestGroup> rg(new RequestGroup(option));
// If piece hash is specified in the metalink, SharedHandle<DownloadContext> dctx;
// make segment size equal to piece hash size. if(mes.size() == 1) {
size_t pieceLength; SharedHandle<MetalinkEntry> entry = mes[0];
_logger->info(MSG_METALINK_QUEUEING, entry->getPath().c_str());
entry->reorderResourcesByPriority();
std::deque<std::string> uris;
std::for_each(entry->resources.begin(), entry->resources.end(),
AccumulateNonP2PUrl(uris));
// If piece hash is specified in the metalink,
// make segment size equal to piece hash size.
size_t pieceLength;
#ifdef ENABLE_MESSAGE_DIGEST #ifdef ENABLE_MESSAGE_DIGEST
if(entry->chunkChecksum.isNull()) { if(entry->chunkChecksum.isNull()) {
pieceLength = option->getAsInt(PREF_SEGMENT_SIZE); pieceLength = option->getAsInt(PREF_SEGMENT_SIZE);
} else { } else {
pieceLength = entry->chunkChecksum->getChecksumLength(); pieceLength = entry->chunkChecksum->getChecksumLength();
}
#else
pieceLength = option->getAsInt(PREF_SEGMENT_SIZE);
#endif // ENABLE_MESSAGE_DIGEST
SharedHandle<DownloadContext> dctx
(new DownloadContext
(pieceLength,
entry->getLength(),
util::applyDir(option->get(PREF_DIR), entry->file->getPath())));
dctx->setDir(option->get(PREF_DIR));
dctx->getFirstFileEntry()->setUris(uris);
if(option->getAsBool(PREF_METALINK_ENABLE_UNIQUE_PROTOCOL)) {
dctx->getFirstFileEntry()->disableSingleHostMultiConnection();
}
#ifdef ENABLE_MESSAGE_DIGEST
if(entry->chunkChecksum.isNull()) {
if(!entry->checksum.isNull()) {
dctx->setChecksum(entry->checksum->getMessageDigest());
dctx->setChecksumHashAlgo(entry->checksum->getAlgo());
} }
} else { #else
dctx->setPieceHashes(entry->chunkChecksum->getChecksums().begin(), pieceLength = option->getAsInt(PREF_SEGMENT_SIZE);
entry->chunkChecksum->getChecksums().end());
dctx->setPieceHashAlgo(entry->chunkChecksum->getAlgo());
}
#endif // ENABLE_MESSAGE_DIGEST #endif // ENABLE_MESSAGE_DIGEST
dctx->setSignature(entry->getSignature()); dctx.reset(new DownloadContext
(pieceLength,
entry->getLength(),
util::applyDir(option->get(PREF_DIR),
entry->file->getPath())));
dctx->getFirstFileEntry()->setUris(uris);
if(option->getAsBool(PREF_METALINK_ENABLE_UNIQUE_PROTOCOL)) {
dctx->getFirstFileEntry()->disableSingleHostMultiConnection();
}
#ifdef ENABLE_MESSAGE_DIGEST
if(entry->chunkChecksum.isNull()) {
if(!entry->checksum.isNull()) {
dctx->setChecksum(entry->checksum->getMessageDigest());
dctx->setChecksumHashAlgo(entry->checksum->getAlgo());
}
} else {
dctx->setPieceHashes(entry->chunkChecksum->getChecksums().begin(),
entry->chunkChecksum->getChecksums().end());
dctx->setPieceHashAlgo(entry->chunkChecksum->getAlgo());
}
#endif // ENABLE_MESSAGE_DIGEST
dctx->setSignature(entry->getSignature());
rg->setNumConcurrentCommand
(entry->maxConnections < 0 ?
option->getAsInt(PREF_METALINK_SERVERS) :
std::min(option->getAsInt(PREF_METALINK_SERVERS),
static_cast<int32_t>(entry->maxConnections)));
} else {
dctx.reset(new DownloadContext());
// piece length is overridden by the one in torrent file.
dctx->setPieceLength(option->getAsInt(PREF_SEGMENT_SIZE));
std::vector<SharedHandle<FileEntry> > fileEntries;
off_t offset = 0;
for(std::deque<SharedHandle<MetalinkEntry> >::const_iterator i =
entries.begin(); i != entries.end(); ++i) {
_logger->info("Metalink: Queueing %s for download as a member.",
(*i)->getPath().c_str());
_logger->debug("originalName = %s", (*i)->metaurls[0]->name.c_str());
(*i)->reorderResourcesByPriority();
std::deque<std::string> uris;
std::for_each((*i)->resources.begin(), (*i)->resources.end(),
AccumulateNonP2PUrl(uris));
SharedHandle<FileEntry> fe
(new FileEntry
(util::applyDir(option->get(PREF_DIR), (*i)->file->getPath()),
(*i)->file->getLength(), offset, uris));
if(option->getAsBool(PREF_METALINK_ENABLE_UNIQUE_PROTOCOL)) {
fe->disableSingleHostMultiConnection();
}
fe->setOriginalName((*i)->metaurls[0]->name);
fileEntries.push_back(fe);
offset += (*i)->file->getLength();
}
dctx->setFileEntries(fileEntries.begin(), fileEntries.end());
rg->setNumConcurrentCommand(option->getAsInt(PREF_METALINK_SERVERS));
}
dctx->setDir(option->get(PREF_DIR));
rg->setDownloadContext(dctx); rg->setDownloadContext(dctx);
rg->setNumConcurrentCommand // remove "metalink" from Accept Type list to avoid loop in
(entry->maxConnections < 0 ? // tranparent metalink
option->getAsInt(PREF_METALINK_SERVERS) :
std::min(option->getAsInt(PREF_METALINK_SERVERS),
static_cast<int32_t>(entry->maxConnections)));
// remove "metalink" from Accept Type list to avoid loop in tranparent
// metalink
removeMetalinkContentTypes(rg); removeMetalinkContentTypes(rg);
#ifdef ENABLE_BITTORRENT #ifdef ENABLE_BITTORRENT
// Inject depenency between rg and torrentRg here if torrentRg.isNull() == false // Inject depenency between rg and torrentRg here if
// torrentRg.isNull() == false
if(!torrentRg.isNull()) { if(!torrentRg.isNull()) {
SharedHandle<Dependency> dep(new BtDependency(rg, torrentRg)); SharedHandle<Dependency> dep(new BtDependency(rg, torrentRg));
rg->dependsOn(dep); rg->dependsOn(dep);
torrentRg->belongsTo(rg->getGID()); torrentRg->belongsTo(rg->getGID());
} }
#endif // ENABLE_BITTORRENT #endif // ENABLE_BITTORRENT

View file

@ -54,7 +54,7 @@ private:
void void
createRequestGroup(std::deque<SharedHandle<RequestGroup> >& groups, createRequestGroup(std::deque<SharedHandle<RequestGroup> >& groups,
std::deque<SharedHandle<MetalinkEntry> > entries, const std::deque<SharedHandle<MetalinkEntry> >& entries,
const SharedHandle<Option>& option); const SharedHandle<Option>& option);
public: public:
Metalink2RequestGroup(); Metalink2RequestGroup();

View file

@ -51,7 +51,7 @@
namespace aria2 { namespace aria2 {
MetalinkEntry::MetalinkEntry(): MetalinkEntry::MetalinkEntry():
file(0), sizeKnown(false),
maxConnections(-1) maxConnections(-1)
{} {}
@ -94,7 +94,7 @@ MetalinkEntry& MetalinkEntry::operator=(const MetalinkEntry& metalinkEntry)
return *this; return *this;
} }
std::string MetalinkEntry::getPath() const const std::string& MetalinkEntry::getPath() const
{ {
return file->getPath(); return file->getPath();
} }

View file

@ -61,6 +61,8 @@ public:
std::string version; std::string version;
std::vector<std::string> languages; std::vector<std::string> languages;
std::vector<std::string> oses; std::vector<std::string> oses;
// True if size is specified in Metalink document.
bool sizeKnown;
std::deque<SharedHandle<MetalinkResource> > resources; std::deque<SharedHandle<MetalinkResource> > resources;
std::vector<SharedHandle<MetalinkMetaurl> > metaurls; std::vector<SharedHandle<MetalinkMetaurl> > metaurls;
int maxConnections; // Metalink3Spec int maxConnections; // Metalink3Spec
@ -77,7 +79,7 @@ public:
MetalinkEntry& operator=(const MetalinkEntry& metalinkEntry); MetalinkEntry& operator=(const MetalinkEntry& metalinkEntry);
std::string getPath() const; const std::string& getPath() const;
uint64_t getLength() const; uint64_t getLength() const;

View file

@ -41,6 +41,7 @@
#include "prefs.h" #include "prefs.h"
#include "DlAbortEx.h" #include "DlAbortEx.h"
#include "BinaryStream.h" #include "BinaryStream.h"
#include "MetalinkMetaurl.h"
namespace aria2 { namespace aria2 {
@ -79,4 +80,41 @@ void MetalinkHelper::query
option->get(PREF_METALINK_OS)); option->get(PREF_METALINK_OS));
} }
void MetalinkHelper::groupEntryByMetaurlName
(std::vector<
std::pair<std::string, std::vector<SharedHandle<MetalinkEntry> > > >& result,
const std::vector<SharedHandle<MetalinkEntry> >& entries)
{
for(std::vector<SharedHandle<MetalinkEntry> >::const_iterator eiter =
entries.begin(); eiter != entries.end(); ++eiter) {
if((*eiter)->metaurls.empty()) {
std::pair<std::string, std::vector<SharedHandle<MetalinkEntry> > > p;
p.second.push_back(*eiter);
result.push_back(p);
} else {
std::vector<
std::pair<std::string,
std::vector<SharedHandle<MetalinkEntry> > > >::iterator i =
result.begin();
if((*eiter)->metaurls[0]->name.empty() ||
!(*eiter)->sizeKnown) {
i = result.end();
}
for(; i != result.end(); ++i) {
if((*i).first == (*eiter)->metaurls[0]->url &&
!(*i).second[0]->metaurls[0]->name.empty()) {
(*i).second.push_back(*eiter);
break;
}
}
if(i == result.end()) {
std::pair<std::string, std::vector<SharedHandle<MetalinkEntry> > > p;
p.first = (*eiter)->metaurls[0]->url;
p.second.push_back(*eiter);
result.push_back(p);
}
}
}
}
} // namespace aria2 } // namespace aria2

View file

@ -36,9 +36,12 @@
#define _D_METALINK_HELPER_H_ #define _D_METALINK_HELPER_H_
#include "common.h" #include "common.h"
#include "SharedHandle.h"
#include <string> #include <string>
#include <deque> #include <deque>
#include <vector>
#include "SharedHandle.h"
namespace aria2 { namespace aria2 {
@ -65,6 +68,11 @@ public:
static void parseAndQuery static void parseAndQuery
(std::deque<SharedHandle<MetalinkEntry> >& result, (std::deque<SharedHandle<MetalinkEntry> >& result,
const SharedHandle<BinaryStream>& binaryStream, const Option* option); const SharedHandle<BinaryStream>& binaryStream, const Option* option);
static void groupEntryByMetaurlName
(std::vector<
std::pair<std::string, std::vector<SharedHandle<MetalinkEntry> > > >& result,
const std::vector<SharedHandle<MetalinkEntry> >& entries);
}; };
} // namespace aria2 } // namespace aria2

View file

@ -42,4 +42,9 @@ const std::string MetalinkMetaurl::MEDIATYPE_TORRENT("torrent");
MetalinkMetaurl::MetalinkMetaurl(): MetalinkMetaurl::MetalinkMetaurl():
priority(MetalinkResource::getLowestPriority()) {} priority(MetalinkResource::getLowestPriority()) {}
MetalinkMetaurl::MetalinkMetaurl
(const std::string& url, const std::string& mediatype,
const std::string& name, int priority):
url(url), mediatype(mediatype), name(name), priority(priority) {}
} // namespace aria2 } // namespace aria2

View file

@ -45,11 +45,14 @@ class MetalinkMetaurl {
public: public:
std::string url; std::string url;
std::string mediatype; std::string mediatype;
int priority;
std::string name; std::string name;
int priority;
MetalinkMetaurl(); MetalinkMetaurl();
MetalinkMetaurl(const std::string& url, const std::string& mediatype,
const std::string& name, int priority);
static const std::string MEDIATYPE_TORRENT; static const std::string MEDIATYPE_TORRENT;
}; };

View file

@ -100,6 +100,7 @@ void MetalinkParserController::setFileLengthOfEntry(uint64_t length)
} else { } else {
_tEntry->file->setLength(length); _tEntry->file->setLength(length);
} }
_tEntry->sizeKnown = true;
} }
void MetalinkParserController::setVersionOfEntry(const std::string& version) void MetalinkParserController::setVersionOfEntry(const std::string& version)

View file

@ -205,6 +205,8 @@ void RequestGroup::createInitialCommand
SharedHandle<BtRegistry> btRegistry = e->getBtRegistry(); SharedHandle<BtRegistry> btRegistry = e->getBtRegistry();
if(!btRegistry->getDownloadContext if(!btRegistry->getDownloadContext
(torrentAttrs[bittorrent::INFO_HASH].s()).isNull()) { (torrentAttrs[bittorrent::INFO_HASH].s()).isNull()) {
// TODO If metadataGetMode == false and each FileEntry has
// URI, then go without BT.
throw DOWNLOAD_FAILURE_EXCEPTION throw DOWNLOAD_FAILURE_EXCEPTION
(StringFormat (StringFormat
("InfoHash %s is already registered.", ("InfoHash %s is already registered.",
@ -284,14 +286,7 @@ void RequestGroup::createInitialCommand
return; return;
} }
removeDefunctControlFile(progressInfoFile);
// Remove the control file if download file doesn't exist
if(progressInfoFile->exists() && !_pieceStorage->getDiskAdaptor()->fileExists()) {
progressInfoFile->removeFile();
_logger->notice(MSG_REMOVED_DEFUNCT_CONTROL_FILE,
progressInfoFile->getFilename().c_str(),
_downloadContext->getBasePath().c_str());
}
{ {
uint64_t actualFileSize = _pieceStorage->getDiskAdaptor()->size(); uint64_t actualFileSize = _pieceStorage->getDiskAdaptor()->size();
if(actualFileSize == _downloadContext->getTotalLength()) { if(actualFileSize == _downloadContext->getTotalLength()) {
@ -372,42 +367,80 @@ void RequestGroup::createInitialCommand
} }
} }
#endif // ENABLE_BITTORRENT #endif // ENABLE_BITTORRENT
// TODO Currently, BitTorrent+WEB-Seeding is only way to download if(_downloadContext->getFileEntries().size() == 1) {
// multiple files in one RequestGroup. In this context, we don't // TODO I assume here when totallength is set to DownloadContext and it is
// have BitTorrent, so add assertion here. This situation will be // not 0, then filepath is also set DownloadContext correctly....
// changed if Metalink spec is formalized to support multi-file if(_option->getAsBool(PREF_DRY_RUN) ||
// torrent. _downloadContext->getTotalLength() == 0) {
assert(_downloadContext->getFileEntries().size() == 1); createNextCommand(commands, e, 1);
// TODO I assume here when totallength is set to DownloadContext and it is }else {
// not 0, then filepath is also set DownloadContext correctly.... if(e->_requestGroupMan->isSameFileBeingDownloaded(this)) {
if(_option->getAsBool(PREF_DRY_RUN) || throw DOWNLOAD_FAILURE_EXCEPTION
_downloadContext->getTotalLength() == 0) { (StringFormat(EX_DUPLICATE_FILE_DOWNLOAD,
createNextCommand(commands, e, 1); _downloadContext->getBasePath().c_str()).str());
}else { }
adjustFilename
(SharedHandle<BtProgressInfoFile>(new DefaultBtProgressInfoFile
(_downloadContext,
SharedHandle<PieceStorage>(),
_option.get())));
initPieceStorage();
BtProgressInfoFileHandle infoFile
(new DefaultBtProgressInfoFile(_downloadContext, _pieceStorage,
_option.get()));
if(!infoFile->exists() && downloadFinishedByFileLength()) {
_pieceStorage->markAllPiecesDone();
_logger->notice(MSG_DOWNLOAD_ALREADY_COMPLETED,
_gid, _downloadContext->getBasePath().c_str());
} else {
loadAndOpenFile(infoFile);
SharedHandle<CheckIntegrityEntry> checkIntegrityEntry
(new StreamCheckIntegrityEntry(this));
processCheckIntegrityEntry(commands, checkIntegrityEntry, e);
}
}
} else {
// In this context, multiple FileEntry objects are in
// DownloadContext.
if(e->_requestGroupMan->isSameFileBeingDownloaded(this)) { if(e->_requestGroupMan->isSameFileBeingDownloaded(this)) {
throw DOWNLOAD_FAILURE_EXCEPTION throw DOWNLOAD_FAILURE_EXCEPTION
(StringFormat(EX_DUPLICATE_FILE_DOWNLOAD, (StringFormat(EX_DUPLICATE_FILE_DOWNLOAD,
_downloadContext->getBasePath().c_str()).str()); _downloadContext->getBasePath().c_str()).str());
} }
adjustFilename
(SharedHandle<BtProgressInfoFile>(new DefaultBtProgressInfoFile
(_downloadContext,
SharedHandle<PieceStorage>(),
_option.get())));
initPieceStorage(); initPieceStorage();
BtProgressInfoFileHandle infoFile if(_downloadContext->getFileEntries().size() > 1) {
(new DefaultBtProgressInfoFile(_downloadContext, _pieceStorage, _pieceStorage->setupFileFilter();
_option.get()));
if(!infoFile->exists() && downloadFinishedByFileLength()) {
_pieceStorage->markAllPiecesDone();
_logger->notice(MSG_DOWNLOAD_ALREADY_COMPLETED,
_gid, _downloadContext->getBasePath().c_str());
} else {
loadAndOpenFile(infoFile);
SharedHandle<CheckIntegrityEntry> checkIntegrityEntry
(new StreamCheckIntegrityEntry(this));
processCheckIntegrityEntry(commands, checkIntegrityEntry, e);
} }
SharedHandle<DefaultBtProgressInfoFile> progressInfoFile
(new DefaultBtProgressInfoFile(_downloadContext,
_pieceStorage,
_option.get()));
removeDefunctControlFile(progressInfoFile);
// Call Load, Save and file allocation command here
if(progressInfoFile->exists()) {
// load .aria2 file if it exists.
progressInfoFile->load();
_pieceStorage->getDiskAdaptor()->openFile();
} else {
if(_pieceStorage->getDiskAdaptor()->fileExists()) {
if(!_option->getAsBool(PREF_CHECK_INTEGRITY) &&
!_option->getAsBool(PREF_ALLOW_OVERWRITE)) {
// TODO we need this->haltRequested = true?
throw DOWNLOAD_FAILURE_EXCEPTION
(StringFormat
(MSG_FILE_ALREADY_EXISTS,
_downloadContext->getBasePath().c_str()).str());
} else {
_pieceStorage->getDiskAdaptor()->openFile();
}
} else {
_pieceStorage->getDiskAdaptor()->openFile();
}
}
_progressInfoFile = progressInfoFile;
SharedHandle<CheckIntegrityEntry> checkIntegrityEntry
(new StreamCheckIntegrityEntry(this));
processCheckIntegrityEntry(commands, checkIntegrityEntry, e);
} }
} }
@ -543,6 +576,19 @@ void RequestGroup::adjustFilename
} }
} }
void RequestGroup::removeDefunctControlFile
(const SharedHandle<BtProgressInfoFile>& progressInfoFile)
{
// Remove the control file if download file doesn't exist
if(progressInfoFile->exists() &&
!_pieceStorage->getDiskAdaptor()->fileExists()) {
progressInfoFile->removeFile();
_logger->notice(MSG_REMOVED_DEFUNCT_CONTROL_FILE,
progressInfoFile->getFilename().c_str(),
_downloadContext->getBasePath().c_str());
}
}
void RequestGroup::loadAndOpenFile(const BtProgressInfoFileHandle& progressInfoFile) void RequestGroup::loadAndOpenFile(const BtProgressInfoFileHandle& progressInfoFile)
{ {
try { try {
@ -550,14 +596,7 @@ void RequestGroup::loadAndOpenFile(const BtProgressInfoFileHandle& progressInfoF
_pieceStorage->getDiskAdaptor()->initAndOpenFile(); _pieceStorage->getDiskAdaptor()->initAndOpenFile();
return; return;
} }
// Remove the control file if download file doesn't exist removeDefunctControlFile(progressInfoFile);
if(progressInfoFile->exists() && !_pieceStorage->getDiskAdaptor()->fileExists()) {
progressInfoFile->removeFile();
_logger->notice(MSG_REMOVED_DEFUNCT_CONTROL_FILE,
progressInfoFile->getFilename().c_str(),
_downloadContext->getBasePath().c_str());
}
if(progressInfoFile->exists()) { if(progressInfoFile->exists()) {
progressInfoFile->load(); progressInfoFile->load();
_pieceStorage->getDiskAdaptor()->openExistingFile(); _pieceStorage->getDiskAdaptor()->openExistingFile();

View file

@ -180,6 +180,9 @@ private:
// _uriResults, then last result code is returned. Otherwise // _uriResults, then last result code is returned. Otherwise
// returns downloadresultcode::UNKNOWN_ERROR. // returns downloadresultcode::UNKNOWN_ERROR.
downloadresultcode::RESULT downloadResult() const; downloadresultcode::RESULT downloadResult() const;
void removeDefunctControlFile
(const SharedHandle<BtProgressInfoFile>& progressInfoFile);
public: public:
// The copy of option is stored in RequestGroup object. // The copy of option is stored in RequestGroup object.
RequestGroup(const SharedHandle<Option>& option); RequestGroup(const SharedHandle<Option>& option);

View file

@ -263,6 +263,7 @@ static void extractFileEntries
(new FileEntry(util::applyDir(ctx->getDir(), path), (new FileEntry(util::applyDir(ctx->getDir(), path),
fileLengthData.i(), fileLengthData.i(),
offset, uris)); offset, uris));
fileEntry->setOriginalName(path);
fileEntries.push_back(fileEntry); fileEntries.push_back(fileEntry);
offset += fileEntry->getLength(); offset += fileEntry->getLength();
} }
@ -291,6 +292,7 @@ static void extractFileEntries
SharedHandle<FileEntry> fileEntry SharedHandle<FileEntry> fileEntry
(new FileEntry(util::applyDir(ctx->getDir(), name),totalLength, 0, (new FileEntry(util::applyDir(ctx->getDir(), name),totalLength, 0,
uris)); uris));
fileEntry->setOriginalName(name);
fileEntries.push_back(fileEntry); fileEntries.push_back(fileEntry);
} }
ctx->setFileEntries(fileEntries.begin(), fileEntries.end()); ctx->setFileEntries(fileEntries.begin(), fileEntries.end());

View file

@ -168,6 +168,8 @@ void BittorrentHelperTest::testGetFileEntries() {
SharedHandle<FileEntry> fileEntry1 = *itr; SharedHandle<FileEntry> fileEntry1 = *itr;
CPPUNIT_ASSERT_EQUAL(std::string("./aria2-test/aria2/src/aria2c"), CPPUNIT_ASSERT_EQUAL(std::string("./aria2-test/aria2/src/aria2c"),
fileEntry1->getPath()); fileEntry1->getPath());
CPPUNIT_ASSERT_EQUAL(std::string("aria2-test/aria2/src/aria2c"),
fileEntry1->getOriginalName());
itr++; itr++;
SharedHandle<FileEntry> fileEntry2 = *itr; SharedHandle<FileEntry> fileEntry2 = *itr;
CPPUNIT_ASSERT_EQUAL(std::string("./aria2-test/aria2-0.2.2.tar.bz2"), CPPUNIT_ASSERT_EQUAL(std::string("./aria2-test/aria2-0.2.2.tar.bz2"),
@ -186,6 +188,8 @@ void BittorrentHelperTest::testGetFileEntriesSingle() {
SharedHandle<FileEntry> fileEntry1 = *itr; SharedHandle<FileEntry> fileEntry1 = *itr;
CPPUNIT_ASSERT_EQUAL(std::string("./aria2-0.8.2.tar.bz2"), CPPUNIT_ASSERT_EQUAL(std::string("./aria2-0.8.2.tar.bz2"),
fileEntry1->getPath()); fileEntry1->getPath());
CPPUNIT_ASSERT_EQUAL(std::string("aria2-0.8.2.tar.bz2"),
fileEntry1->getOriginalName());
} }
void BittorrentHelperTest::testGetTotalLength() { void BittorrentHelperTest::testGetTotalLength() {

View file

@ -23,6 +23,9 @@ class BtDependencyTest:public CppUnit::TestFixture {
CPPUNIT_TEST_SUITE(BtDependencyTest); CPPUNIT_TEST_SUITE(BtDependencyTest);
CPPUNIT_TEST(testResolve); CPPUNIT_TEST(testResolve);
CPPUNIT_TEST(testResolve_originalNameNoMatch);
CPPUNIT_TEST(testResolve_singleFileWithoutOriginalName);
CPPUNIT_TEST(testResolve_multiFile);
CPPUNIT_TEST(testResolve_metadata); CPPUNIT_TEST(testResolve_metadata);
CPPUNIT_TEST(testResolve_loadError); CPPUNIT_TEST(testResolve_loadError);
CPPUNIT_TEST(testResolve_dependeeFailure); CPPUNIT_TEST(testResolve_dependeeFailure);
@ -37,7 +40,9 @@ class BtDependencyTest:public CppUnit::TestFixture {
dctx->setDir("/tmp"); dctx->setDir("/tmp");
std::deque<std::string> uris; std::deque<std::string> uris;
uris.push_back("http://localhost/outfile.path"); uris.push_back("http://localhost/outfile.path");
dctx->getFirstFileEntry()->setUris(uris); SharedHandle<FileEntry> fileEntry = dctx->getFirstFileEntry();
fileEntry->setUris(uris);
fileEntry->setOriginalName("aria2-0.8.2.tar.bz2");
dependant->setDownloadContext(dctx); dependant->setDownloadContext(dctx);
return dependant; return dependant;
} }
@ -65,6 +70,9 @@ public:
} }
void testResolve(); void testResolve();
void testResolve_originalNameNoMatch();
void testResolve_singleFileWithoutOriginalName();
void testResolve_multiFile();
void testResolve_metadata(); void testResolve_metadata();
void testResolve_loadError(); void testResolve_loadError();
void testResolve_dependeeFailure(); void testResolve_dependeeFailure();
@ -93,6 +101,64 @@ void BtDependencyTest::testResolve()
CPPUNIT_ASSERT_EQUAL(std::string("/tmp/outfile.path"), CPPUNIT_ASSERT_EQUAL(std::string("/tmp/outfile.path"),
firstFileEntry->getPath()); firstFileEntry->getPath());
CPPUNIT_ASSERT_EQUAL((size_t)1, firstFileEntry->getRemainingUris().size()); CPPUNIT_ASSERT_EQUAL((size_t)1, firstFileEntry->getRemainingUris().size());
CPPUNIT_ASSERT(firstFileEntry->isRequested());
}
void BtDependencyTest::testResolve_originalNameNoMatch()
{
std::string filename = "single.torrent";
SharedHandle<RequestGroup> dependant = createDependant(_option);
dependant->getDownloadContext()->getFirstFileEntry()->setOriginalName
("aria2-1.1.0.tar.bz2");
SharedHandle<RequestGroup> dependee =
createDependee(_option, filename, File(filename).size());
dependee->getPieceStorage()->markAllPiecesDone();
BtDependency dep(dependant, dependee);
CPPUNIT_ASSERT(dep.resolve());
CPPUNIT_ASSERT(!dependant->getDownloadContext()->hasAttribute
(bittorrent::BITTORRENT));
}
void BtDependencyTest::testResolve_singleFileWithoutOriginalName()
{
std::string filename = "single.torrent";
SharedHandle<RequestGroup> dependant = createDependant(_option);
dependant->getDownloadContext()->getFirstFileEntry()->setOriginalName("");
SharedHandle<RequestGroup> dependee =
createDependee(_option, filename, File(filename).size());
dependee->getPieceStorage()->markAllPiecesDone();
BtDependency dep(dependant, dependee);
CPPUNIT_ASSERT(dep.resolve());
CPPUNIT_ASSERT(dependant->getDownloadContext()->hasAttribute
(bittorrent::BITTORRENT));
}
void BtDependencyTest::testResolve_multiFile()
{
std::string filename = "test.torrent";
SharedHandle<RequestGroup> dependant = createDependant(_option);
dependant->getDownloadContext()->getFirstFileEntry()->setOriginalName
("aria2-test/aria2/src/aria2c");
SharedHandle<RequestGroup> dependee =
createDependee(_option, filename, File(filename).size());
dependee->getPieceStorage()->markAllPiecesDone();
BtDependency dep(dependant, dependee);
CPPUNIT_ASSERT(dep.resolve());
CPPUNIT_ASSERT(dependant->getDownloadContext()->hasAttribute
(bittorrent::BITTORRENT));
const std::vector<SharedHandle<FileEntry> >& fileEntries =
dependant->getDownloadContext()->getFileEntries();
CPPUNIT_ASSERT_EQUAL(std::string("/tmp/outfile.path"),
fileEntries[0]->getPath());
CPPUNIT_ASSERT(fileEntries[0]->isRequested());
CPPUNIT_ASSERT_EQUAL(std::string("/tmp/aria2-test/aria2-0.2.2.tar.bz2"),
fileEntries[1]->getPath());
CPPUNIT_ASSERT(!fileEntries[1]->isRequested());
} }
void BtDependencyTest::testResolve_metadata() void BtDependencyTest::testResolve_metadata()
@ -121,6 +187,8 @@ void BtDependencyTest::testResolve_metadata()
CPPUNIT_ASSERT_EQUAL CPPUNIT_ASSERT_EQUAL
(std::string("cd41c7fdddfd034a15a04d7ff881216e01c4ceaf"), (std::string("cd41c7fdddfd034a15a04d7ff881216e01c4ceaf"),
bittorrent::getInfoHashString(dependant->getDownloadContext())); bittorrent::getInfoHashString(dependant->getDownloadContext()));
CPPUNIT_ASSERT
(dependant->getDownloadContext()->getFirstFileEntry()->isRequested());
} }
void BtDependencyTest::testResolve_loadError() void BtDependencyTest::testResolve_loadError()

View file

@ -1,8 +1,11 @@
#include "MetalinkHelper.h" #include "MetalinkHelper.h"
#include <cppunit/extensions/HelperMacros.h>
#include "MetalinkEntry.h" #include "MetalinkEntry.h"
#include "Option.h" #include "Option.h"
#include "prefs.h" #include "prefs.h"
#include <cppunit/extensions/HelperMacros.h> #include "MetalinkMetaurl.h"
namespace aria2 { namespace aria2 {
@ -11,16 +14,14 @@ class MetalinkHelperTest:public CppUnit::TestFixture {
CPPUNIT_TEST_SUITE(MetalinkHelperTest); CPPUNIT_TEST_SUITE(MetalinkHelperTest);
CPPUNIT_TEST(testParseAndQuery); CPPUNIT_TEST(testParseAndQuery);
CPPUNIT_TEST(testParseAndQuery_version); CPPUNIT_TEST(testParseAndQuery_version);
CPPUNIT_TEST(testGroupEntryByMetaurlName);
CPPUNIT_TEST_SUITE_END(); CPPUNIT_TEST_SUITE_END();
private: private:
public: public:
void setUp() {}
void tearDown() {}
void testParseAndQuery(); void testParseAndQuery();
void testParseAndQuery_version(); void testParseAndQuery_version();
void testGroupEntryByMetaurlName();
}; };
@ -45,4 +46,70 @@ void MetalinkHelperTest::testParseAndQuery_version()
CPPUNIT_ASSERT_EQUAL(std::string("aria2-0.5.1.tar.bz2"), entry->getPath()); CPPUNIT_ASSERT_EQUAL(std::string("aria2-0.5.1.tar.bz2"), entry->getPath());
} }
void MetalinkHelperTest::testGroupEntryByMetaurlName()
{
std::vector<SharedHandle<MetalinkEntry> > entries;
SharedHandle<MetalinkEntry> e1(new MetalinkEntry());
e1->version = "1";
e1->sizeKnown = true;
// no name
e1->metaurls.push_back
(SharedHandle<MetalinkMetaurl>
(new MetalinkMetaurl("http://meta1", "torrent", "", 1)));
SharedHandle<MetalinkEntry> e2(new MetalinkEntry());
e2->version = "2";
e2->sizeKnown = true;
SharedHandle<MetalinkEntry> e3(new MetalinkEntry());
e3->version = "3";
e3->sizeKnown = true;
e3->metaurls.push_back
(SharedHandle<MetalinkMetaurl>
(new MetalinkMetaurl("http://meta2", "torrent", "f3", 1)));
SharedHandle<MetalinkEntry> e4(new MetalinkEntry());
e4->version = "4";
e4->sizeKnown = true;
e4->metaurls.push_back
(SharedHandle<MetalinkMetaurl>
(new MetalinkMetaurl("http://meta1", "torrent", "f4", 1)));
SharedHandle<MetalinkEntry> e5(new MetalinkEntry());
e5->version = "5";
// no size
e5->metaurls.push_back
(SharedHandle<MetalinkMetaurl>
(new MetalinkMetaurl("http://meta1", "torrent", "f5", 1)));
SharedHandle<MetalinkEntry> e6(new MetalinkEntry());
e6->version = "6";
e6->sizeKnown = true;
e6->metaurls.push_back
(SharedHandle<MetalinkMetaurl>
(new MetalinkMetaurl("http://meta1", "torrent", "f6", 1)));
entries.push_back(e1);
entries.push_back(e2);
entries.push_back(e3);
entries.push_back(e4);
entries.push_back(e5);
entries.push_back(e6);
std::vector<std::pair<std::string,
std::vector<SharedHandle<MetalinkEntry> > > > result;
MetalinkHelper::groupEntryByMetaurlName(result, entries);
CPPUNIT_ASSERT_EQUAL(std::string("http://meta1"), result[0].first);
CPPUNIT_ASSERT_EQUAL(std::string("1"), result[0].second[0]->version);
CPPUNIT_ASSERT_EQUAL(std::string(""), result[1].first);
CPPUNIT_ASSERT_EQUAL(std::string("2"), result[1].second[0]->version);
CPPUNIT_ASSERT_EQUAL(std::string("http://meta2"), result[2].first);
CPPUNIT_ASSERT_EQUAL(std::string("3"), result[2].second[0]->version);
CPPUNIT_ASSERT_EQUAL(std::string("http://meta1"), result[3].first);
CPPUNIT_ASSERT_EQUAL(std::string("4"), result[3].second[0]->version);
CPPUNIT_ASSERT_EQUAL(std::string("6"), result[3].second[1]->version);
}
} // namespace aria2 } // namespace aria2