2006-04-16 Tatsuhiro Tsujikawa <tujikawa at rednoah dot com>

* src/TorrentConsoleDownloadEngine.cc
	(onPartialDownloadingCompletes): Renamed as
	onSelectiveDownloadingCompletes.
	(onSelectiveDownloadingCompletes): Updated message.

	To add DiskAdaptor which abstract DiskWriter family:

	* src/PeerConnection.cc: Use diskAdaptor instead of diskWriter.
	* src/PeerInteractionCommand.cc: Use diskAdaptor instead of 
diskWriter.
	* src/BitfieldMan.cc (isAllBitSet): bug fix.
	* src/TorrentMan.cc : Included CopyDiskAdaptor.h, 
DirectDiskAdaptor.h,
	MultiDiskAdaptor.h.
	(readFileEntry): Changed its arguments.
	(setup): setupDiskWriter is merged into this function.
	(setupDiskWriter): Removed.
	(setFileFilter): New function.
	(readFileEntryFromMetaInfoFile): Updated according to the 
changes
	made in readFileEntry.
	(getFilePath): Removed.
	(getTempFilePath): Removed.
	(getSegmentFilePath): Updated due to the removal of getFilePath.
	(fixFilename): Removed.
	(copySingleFile): Removed.
	(splitMultiFile): Removed.
	(deleteTempFile): Removed.
	(setFileEntriesToDownload): Removed.
	(isPartialDownloadingMode): Renamed as 
isSelectiveDownloadingMode.
	(isSelectiveDownloadingMode): New function.
	(setAllMultiFileRequestedState): Removed.
	(finishPartialDownloadingMode): Renamed as
	finishSelectiveDownloadingMode.
	(finishSelectiveDownloadingMode): New function.
	(getPartialTotalLength): Renamed as getSelectedTotalLength.
	(getSelectedTotalLength): New function.
	(onDownloadComplete): Use diskAdaptor.
	* src/MultiDiskWriter.cc (Constructor): Added the argument 
pieceLength
	(Range): Removed.
	(setMultiFileEntries): Renamed as setFileEntries.
	(setFileEntries): New function.
	* src/MultiDiskWriter.h [DiskWriterEntry](enabled): Removed.
	(pieceLength): New variable.
	* src/main.cc (printDownloadCompeleteMessage): New function.
	(torrentHandler): Use diskAdaptor instead of diskWriter.
	(main): Renamed torrent-show-files to show-files.
	Rewritten file contents listing.
	* src/TorrentMan.h (FileEntry): Removed.
	(multiFileTopDir): Removed.
	(multiFileEntries): Removed.
	(diskWriter): Removed.
	(diskAdaptor): New variable.
	* src/DefaultDiskWriter.h (totalLength): New variable.
	* src/DefaultDiskWriter.cc (initAndOpenFile): Added ftruncate.
	* src/TorrentDownloadEngine.cc (onEndOfRun): Use diskAdaptor 
instead of
	diskWriter.
	* src/TorrentConsoleDownloadEngine.h
	(partialDownloadLengthDiff): Renamed as 
selectedDownloadLengthDiff.
	(partialTotalLength): Renamed as selectedTotalLength.
	* src/AbstractDiskWriter.cc (openFile): If file exists, call
	openExistingFile, otherwise call initAndOpenFile.
	(closeFile): fd > 0, not fd != 0.
	
	* src/prefs.h (PREF_TORRENT_SHOW_FILES): Renamed as 
PREF_SHOW_FILES
	(PREF_SHOW_FILES): New definition.
pull/1/head
Tatsuhiro Tsujikawa 2006-04-16 14:38:19 +00:00
parent 58633887e9
commit 2a84b9de43
23 changed files with 302 additions and 343 deletions

View File

@ -1,3 +1,71 @@
2006-04-16 Tatsuhiro Tsujikawa <tujikawa at rednoah dot com>
* src/TorrentConsoleDownloadEngine.cc
(onPartialDownloadingCompletes): Renamed as
onSelectiveDownloadingCompletes.
(onSelectiveDownloadingCompletes): Updated message.
To add DiskAdaptor which abstract DiskWriter family:
* src/PeerConnection.cc: Use diskAdaptor instead of diskWriter.
* src/PeerInteractionCommand.cc: Use diskAdaptor instead of diskWriter.
* src/BitfieldMan.cc (isAllBitSet): bug fix.
* src/TorrentMan.cc : Included CopyDiskAdaptor.h, DirectDiskAdaptor.h,
MultiDiskAdaptor.h.
(readFileEntry): Changed its arguments.
(setup): setupDiskWriter is merged into this function.
(setupDiskWriter): Removed.
(setFileFilter): New function.
(readFileEntryFromMetaInfoFile): Updated according to the changes
made in readFileEntry.
(getFilePath): Removed.
(getTempFilePath): Removed.
(getSegmentFilePath): Updated due to the removal of getFilePath.
(fixFilename): Removed.
(copySingleFile): Removed.
(splitMultiFile): Removed.
(deleteTempFile): Removed.
(setFileEntriesToDownload): Removed.
(isPartialDownloadingMode): Renamed as isSelectiveDownloadingMode.
(isSelectiveDownloadingMode): New function.
(setAllMultiFileRequestedState): Removed.
(finishPartialDownloadingMode): Renamed as
finishSelectiveDownloadingMode.
(finishSelectiveDownloadingMode): New function.
(getPartialTotalLength): Renamed as getSelectedTotalLength.
(getSelectedTotalLength): New function.
(onDownloadComplete): Use diskAdaptor.
* src/MultiDiskWriter.cc (Constructor): Added the argument pieceLength
(Range): Removed.
(setMultiFileEntries): Renamed as setFileEntries.
(setFileEntries): New function.
* src/MultiDiskWriter.h [DiskWriterEntry](enabled): Removed.
(pieceLength): New variable.
* src/main.cc (printDownloadCompeleteMessage): New function.
(torrentHandler): Use diskAdaptor instead of diskWriter.
(main): Renamed torrent-show-files to show-files.
Rewritten file contents listing.
* src/TorrentMan.h (FileEntry): Removed.
(multiFileTopDir): Removed.
(multiFileEntries): Removed.
(diskWriter): Removed.
(diskAdaptor): New variable.
* src/DefaultDiskWriter.h (totalLength): New variable.
* src/DefaultDiskWriter.cc (initAndOpenFile): Added ftruncate.
* src/TorrentDownloadEngine.cc (onEndOfRun): Use diskAdaptor instead of
diskWriter.
* src/TorrentConsoleDownloadEngine.h
(partialDownloadLengthDiff): Renamed as selectedDownloadLengthDiff.
(partialTotalLength): Renamed as selectedTotalLength.
* src/AbstractDiskWriter.cc (openFile): If file exists, call
openExistingFile, otherwise call initAndOpenFile.
(closeFile): fd > 0, not fd != 0.
* src/prefs.h (PREF_TORRENT_SHOW_FILES): Renamed as PREF_SHOW_FILES
(PREF_SHOW_FILES): New definition.
2006-04-12 Tatsuhiro Tsujikawa <tujikawa at rednoah dot com> 2006-04-12 Tatsuhiro Tsujikawa <tujikawa at rednoah dot com>
To add the ability to download multi torrent into respective files To add the ability to download multi torrent into respective files

12
TODO
View File

@ -11,6 +11,14 @@
* Add port range command-line option * Add port range command-line option
* Add max peers command-line option * Add max peers command-line option
* Distinguish seeder from leecher * Distinguish seeder from leecher
* file selection in multi-file mode
0.4.0 release
* Add selective downloading mode
* try to use ftruncate to allocate file. * try to use ftruncate to allocate file.
* fix TorrentMan::getFilePath() * test all download mode
* Add loggerfactory
* Add SIGTERM signal handler
0.4.1 release
* Add port command-line option
* Add estimated remaining time to normal HTTP/FTP downloading status ouput

View File

@ -36,7 +36,7 @@ AbstractDiskWriter::AbstractDiskWriter():fd(0) {
} }
AbstractDiskWriter::~AbstractDiskWriter() { AbstractDiskWriter::~AbstractDiskWriter() {
if(fd != 0) { if(fd > 0) {
close(fd); close(fd);
} }
#ifdef ENABLE_SHA1DIGEST #ifdef ENABLE_SHA1DIGEST
@ -45,13 +45,16 @@ AbstractDiskWriter::~AbstractDiskWriter() {
} }
void AbstractDiskWriter::openFile(const string& filename) { void AbstractDiskWriter::openFile(const string& filename) {
if((fd = open(filename.c_str(), O_CREAT|O_RDWR, S_IRUSR|S_IWUSR)) < 0) { File f(filename);
throw new DlAbortEx(strerror(errno)); if(f.exists()) {
openExistingFile(filename);
} else {
initAndOpenFile(filename);
} }
} }
void AbstractDiskWriter::closeFile() { void AbstractDiskWriter::closeFile() {
if(fd != 0) { if(fd > 0) {
close(fd); close(fd);
fd = 0; fd = 0;
} }

View File

@ -261,7 +261,7 @@ bool BitfieldMan::unsetBit(int index) {
bool BitfieldMan::isAllBitSet() const { bool BitfieldMan::isAllBitSet() const {
if(filterEnabled) { if(filterEnabled) {
for(int i = 0; i < bitfieldLength-1; i++) { for(int i = 0; i < bitfieldLength; i++) {
if((bitfield[i]&filterBitfield[i]) != filterBitfield[i]) { if((bitfield[i]&filterBitfield[i]) != filterBitfield[i]) {
return false; return false;
} }

View File

@ -23,10 +23,15 @@
#include "DlAbortEx.h" #include "DlAbortEx.h"
#include <errno.h> #include <errno.h>
DefaultDiskWriter::DefaultDiskWriter():AbstractDiskWriter() {} DefaultDiskWriter::DefaultDiskWriter():AbstractDiskWriter(), totalLength(0) {}
DefaultDiskWriter::DefaultDiskWriter(long long int totalLength):AbstractDiskWriter(), totalLength(totalLength) {}
DefaultDiskWriter::~DefaultDiskWriter() {} DefaultDiskWriter::~DefaultDiskWriter() {}
void DefaultDiskWriter::initAndOpenFile(string filename) { void DefaultDiskWriter::initAndOpenFile(string filename) {
createFile(filename); createFile(filename);
if(totalLength > 0) {
ftruncate(fd, totalLength);
}
} }

View File

@ -25,8 +25,11 @@
#include "AbstractDiskWriter.h" #include "AbstractDiskWriter.h"
class DefaultDiskWriter:public AbstractDiskWriter { class DefaultDiskWriter:public AbstractDiskWriter {
private:
long long int totalLength;
public: public:
DefaultDiskWriter(); DefaultDiskWriter();
DefaultDiskWriter(long long int totalLength);
~DefaultDiskWriter(); ~DefaultDiskWriter();
void initAndOpenFile(string filename); void initAndOpenFile(string filename);

42
src/FileEntry.h Normal file
View File

@ -0,0 +1,42 @@
/* <!-- copyright */
/*
* aria2 - a simple utility for downloading files faster
*
* Copyright (C) 2006 Tatsuhiro Tsujikawa
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
/* copyright --> */
#ifndef _D_FILE_ENTRY_H_
#define _D_FILE_ENTRY_H_
#include "common.h"
class FileEntry {
public:
string path;
long long int length;
long long int offset;
bool extracted;
bool requested;
FileEntry(string path, long long int length, long long int offset):
path(path), length(length), offset(offset),
extracted(false), requested(true) {}
~FileEntry() {}
};
typedef deque<FileEntry> FileEntries;
#endif // _D_FILE_ENTRY_H_

View File

@ -79,7 +79,12 @@ SRCS = Socket.cc Socket.h\
TrackerWatcherCommand.cc TrackerWatcherCommand.h\ TrackerWatcherCommand.cc TrackerWatcherCommand.h\
messageDigest.h\ messageDigest.h\
SendMessageQueue.cc SendMessageQueue.h\ SendMessageQueue.cc SendMessageQueue.h\
MultiDiskWriter.cc MultiDiskWriter.h MultiDiskWriter.cc MultiDiskWriter.h\
DiskAdaptor.cc DiskAdaptor.h\
CopyDiskAdaptor.cc CopyDiskAdaptor.h\
DirectDiskAdaptor.cc DirectDiskAdaptor.h\
MultiDiskAdaptor.cc MultiDiskAdaptor.h\
FileEntry.h
noinst_LIBRARIES = libaria2c.a noinst_LIBRARIES = libaria2c.a
libaria2c_a_SOURCES = $(SRCS) libaria2c_a_SOURCES = $(SRCS)
aria2c_LDADD = libaria2c.a @LIBINTL@ @ALLOCA@ @LIBGNUTLS_LIBS@\ aria2c_LDADD = libaria2c.a @LIBINTL@ @ALLOCA@ @LIBGNUTLS_LIBS@\

View File

@ -98,7 +98,9 @@ am__objects_1 = Socket.$(OBJEXT) SocketCore.$(OBJEXT) \
PeerMessage.$(OBJEXT) Piece.$(OBJEXT) RequestSlot.$(OBJEXT) \ PeerMessage.$(OBJEXT) Piece.$(OBJEXT) RequestSlot.$(OBJEXT) \
RequestSlotMan.$(OBJEXT) TorrentAutoSaveCommand.$(OBJEXT) \ RequestSlotMan.$(OBJEXT) TorrentAutoSaveCommand.$(OBJEXT) \
Directory.$(OBJEXT) TrackerWatcherCommand.$(OBJEXT) \ Directory.$(OBJEXT) TrackerWatcherCommand.$(OBJEXT) \
SendMessageQueue.$(OBJEXT) MultiDiskWriter.$(OBJEXT) SendMessageQueue.$(OBJEXT) MultiDiskWriter.$(OBJEXT) \
DiskAdaptor.$(OBJEXT) CopyDiskAdaptor.$(OBJEXT) \
DirectDiskAdaptor.$(OBJEXT) MultiDiskAdaptor.$(OBJEXT)
am_libaria2c_a_OBJECTS = $(am__objects_1) am_libaria2c_a_OBJECTS = $(am__objects_1)
libaria2c_a_OBJECTS = $(am_libaria2c_a_OBJECTS) libaria2c_a_OBJECTS = $(am_libaria2c_a_OBJECTS)
am__installdirs = "$(DESTDIR)$(bindir)" am__installdirs = "$(DESTDIR)$(bindir)"
@ -327,7 +329,12 @@ SRCS = Socket.cc Socket.h\
TrackerWatcherCommand.cc TrackerWatcherCommand.h\ TrackerWatcherCommand.cc TrackerWatcherCommand.h\
messageDigest.h\ messageDigest.h\
SendMessageQueue.cc SendMessageQueue.h\ SendMessageQueue.cc SendMessageQueue.h\
MultiDiskWriter.cc MultiDiskWriter.h MultiDiskWriter.cc MultiDiskWriter.h\
DiskAdaptor.cc DiskAdaptor.h\
CopyDiskAdaptor.cc CopyDiskAdaptor.h\
DirectDiskAdaptor.cc DirectDiskAdaptor.h\
MultiDiskAdaptor.cc MultiDiskAdaptor.h\
FileEntry.h
noinst_LIBRARIES = libaria2c.a noinst_LIBRARIES = libaria2c.a
libaria2c_a_SOURCES = $(SRCS) libaria2c_a_SOURCES = $(SRCS)
@ -420,10 +427,13 @@ distclean-compile:
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/ChunkedEncoding.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/ChunkedEncoding.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/ConsoleDownloadEngine.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/ConsoleDownloadEngine.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/CookieBox.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/CookieBox.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/CopyDiskAdaptor.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/Data.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/Data.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/DefaultDiskWriter.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/DefaultDiskWriter.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/Dictionary.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/Dictionary.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/DirectDiskAdaptor.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/Directory.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/Directory.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/DiskAdaptor.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/DownloadCommand.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/DownloadCommand.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/DownloadEngine.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/DownloadEngine.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/File.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/File.Po@am__quote@
@ -444,6 +454,7 @@ distclean-compile:
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/InitiateConnectionCommandFactory.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/InitiateConnectionCommandFactory.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/List.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/List.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/MetaFileUtil.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/MetaFileUtil.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/MultiDiskAdaptor.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/MultiDiskWriter.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/MultiDiskWriter.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/Option.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/Option.Po@am__quote@
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/Peer.Po@am__quote@ @AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/Peer.Po@am__quote@

View File

@ -24,7 +24,7 @@
#include "Util.h" #include "Util.h"
#include <errno.h> #include <errno.h>
MultiDiskWriter::MultiDiskWriter() { MultiDiskWriter::MultiDiskWriter(int pieceLength):pieceLength(pieceLength) {
#ifdef ENABLE_SHA1DIGEST #ifdef ENABLE_SHA1DIGEST
sha1DigestInit(ctx); sha1DigestInit(ctx);
#endif // ENABLE_SHA1DIGEST #endif // ENABLE_SHA1DIGEST
@ -37,93 +37,50 @@ MultiDiskWriter::~MultiDiskWriter() {
#endif // ENABLE_SHA1DIGEST #endif // ENABLE_SHA1DIGEST
} }
typedef struct {
long long int blockOffset;
long long int length;
} Range;
typedef deque<Range> Ranges;
void MultiDiskWriter::clearEntries() { void MultiDiskWriter::clearEntries() {
for(DiskWriterEntries::iterator itr = diskWriterEntries.begin(); for(DiskWriterEntries::iterator itr = diskWriterEntries.begin();
itr != diskWriterEntries.end(); itr++) { itr != diskWriterEntries.end(); itr++) {
if((*itr)->enabled) {
(*itr)->diskWriter->closeFile();
}
delete *itr; delete *itr;
} }
diskWriterEntries.clear(); diskWriterEntries.clear();
} }
void MultiDiskWriter::setMultiFileEntries(const MultiFileEntries& multiFileEntries, int pieceLength) { void MultiDiskWriter::setFileEntries(const FileEntries& fileEntries) {
clearEntries(); clearEntries();
Ranges ranges; for(FileEntries::const_iterator itr = fileEntries.begin();
for(MultiFileEntries::const_iterator itr = multiFileEntries.begin(); itr != fileEntries.end(); itr++) {
itr != multiFileEntries.end(); itr++) { diskWriterEntries.push_back(new DiskWriterEntry(*itr));
if(itr->requested) {
Range range;
range.blockOffset = (itr->offset/pieceLength)*pieceLength;
range.length = ((itr->offset+itr->length)/pieceLength+
((itr->offset+itr->length)%pieceLength ? 1 : 0))*pieceLength-range.blockOffset;
ranges.push_back(range);
}
}
Ranges::const_iterator ritr = ranges.begin();
for(MultiFileEntries::const_iterator itr = multiFileEntries.begin();
itr != multiFileEntries.end(); itr++) {
DiskWriterEntry* entry;
if(ritr != ranges.end() &&
// !(ritr->blockOffset+ritr->length-1 < itr->offset ||
// itr->offset+itr->length-1 < ritr->blockOffset)) {
itr->offset < ritr->blockOffset+ritr->length &&
ritr->blockOffset < itr->offset+itr->length) {
entry = new DiskWriterEntry(*itr, true);
for(;ritr->blockOffset+ritr->length <= itr->offset+itr->length &&
ritr != ranges.end(); ritr++);
} else {
entry = new DiskWriterEntry(*itr, false);
}
diskWriterEntries.push_back(entry);
} }
} }
void MultiDiskWriter::openFile(const string& filename) { void MultiDiskWriter::openFile(const string& filename) {
for(DiskWriterEntries::iterator itr = diskWriterEntries.begin(); for(DiskWriterEntries::iterator itr = diskWriterEntries.begin();
itr != diskWriterEntries.end(); itr++) { itr != diskWriterEntries.end(); itr++) {
if((*itr)->enabled) {
(*itr)->diskWriter->openFile(filename+"/"+(*itr)->fileEntry.path); (*itr)->diskWriter->openFile(filename+"/"+(*itr)->fileEntry.path);
} }
} }
}
// filename is a directory which is specified by the user in the option. // filename is a directory which is specified by the user in the option.
void MultiDiskWriter::initAndOpenFile(string filename) { void MultiDiskWriter::initAndOpenFile(string filename) {
for(DiskWriterEntries::iterator itr = diskWriterEntries.begin(); for(DiskWriterEntries::iterator itr = diskWriterEntries.begin();
itr != diskWriterEntries.end(); itr++) { itr != diskWriterEntries.end(); itr++) {
if((*itr)->enabled) {
(*itr)->diskWriter->initAndOpenFile(filename+"/"+(*itr)->fileEntry.path); (*itr)->diskWriter->initAndOpenFile(filename+"/"+(*itr)->fileEntry.path);
} }
} }
}
void MultiDiskWriter::closeFile() { void MultiDiskWriter::closeFile() {
for(DiskWriterEntries::iterator itr = diskWriterEntries.begin(); for(DiskWriterEntries::iterator itr = diskWriterEntries.begin();
itr != diskWriterEntries.end(); itr++) { itr != diskWriterEntries.end(); itr++) {
if((*itr)->enabled) {
(*itr)->diskWriter->closeFile(); (*itr)->diskWriter->closeFile();
} }
} }
}
void MultiDiskWriter::openExistingFile(string filename) { void MultiDiskWriter::openExistingFile(string filename) {
for(DiskWriterEntries::iterator itr = diskWriterEntries.begin(); for(DiskWriterEntries::iterator itr = diskWriterEntries.begin();
itr != diskWriterEntries.end(); itr++) { itr != diskWriterEntries.end(); itr++) {
if((*itr)->enabled) {
(*itr)->diskWriter->openExistingFile(filename+"/"+(*itr)->fileEntry.path); (*itr)->diskWriter->openExistingFile(filename+"/"+(*itr)->fileEntry.path);
} }
} }
}
void MultiDiskWriter::writeData(const char* data, int len, long long int position) { void MultiDiskWriter::writeData(const char* data, int len, long long int position) {
long long int offset = position; long long int offset = position;
@ -133,9 +90,6 @@ void MultiDiskWriter::writeData(const char* data, int len, long long int positio
for(DiskWriterEntries::iterator itr = diskWriterEntries.begin(); for(DiskWriterEntries::iterator itr = diskWriterEntries.begin();
itr != diskWriterEntries.end() && rem != 0; itr++) { itr != diskWriterEntries.end() && rem != 0; itr++) {
if(isInRange(*itr, offset) || writing) { if(isInRange(*itr, offset) || writing) {
if(!(*itr)->enabled) {
throw new DlAbortEx("invalid offset or length. offset = %lld", offset);
}
int writeLength = calculateLength(*itr, fileOffset, rem); int writeLength = calculateLength(*itr, fileOffset, rem);
(*itr)->diskWriter->writeData(data+(len-rem), writeLength, fileOffset); (*itr)->diskWriter->writeData(data+(len-rem), writeLength, fileOffset);
rem -= writeLength; rem -= writeLength;
@ -174,9 +128,6 @@ int MultiDiskWriter::readData(char* data, int len, long long int position) {
for(DiskWriterEntries::iterator itr = diskWriterEntries.begin(); for(DiskWriterEntries::iterator itr = diskWriterEntries.begin();
itr != diskWriterEntries.end() && rem != 0; itr++) { itr != diskWriterEntries.end() && rem != 0; itr++) {
if(isInRange(*itr, offset) || reading) { if(isInRange(*itr, offset) || reading) {
if(!(*itr)->enabled) {
throw new DlAbortEx("invalid offset or length. offset = %lld", offset);
}
int readLength = calculateLength((*itr), fileOffset, rem); int readLength = calculateLength((*itr), fileOffset, rem);
totalReadLength += (*itr)->diskWriter->readData(data+(len-rem), readLength, fileOffset); totalReadLength += (*itr)->diskWriter->readData(data+(len-rem), readLength, fileOffset);
rem -= readLength; rem -= readLength;
@ -223,9 +174,6 @@ string MultiDiskWriter::sha1Sum(long long int offset, long long int length) {
for(DiskWriterEntries::iterator itr = diskWriterEntries.begin(); for(DiskWriterEntries::iterator itr = diskWriterEntries.begin();
itr != diskWriterEntries.end() && rem != 0; itr++) { itr != diskWriterEntries.end() && rem != 0; itr++) {
if(isInRange(*itr, offset) || reading) { if(isInRange(*itr, offset) || reading) {
if(!(*itr)->enabled) {
throw new DlAbortEx("invalid offset or length. offset = %lld", offset);
}
int readLength = calculateLength((*itr), fileOffset, rem); int readLength = calculateLength((*itr), fileOffset, rem);
hashUpdate(*itr, fileOffset, readLength); hashUpdate(*itr, fileOffset, readLength);
rem -= readLength; rem -= readLength;

View File

@ -30,18 +30,13 @@ class DiskWriterEntry {
public: public:
FileEntry fileEntry; FileEntry fileEntry;
DiskWriter* diskWriter; DiskWriter* diskWriter;
bool enabled;
public: public:
DiskWriterEntry(const FileEntry& fileEntry, bool enabled):fileEntry(fileEntry), enabled(enabled) { DiskWriterEntry(const FileEntry& fileEntry):fileEntry(fileEntry) {
if(enabled) { diskWriter = new DefaultDiskWriter(this->fileEntry.length);
diskWriter = new DefaultDiskWriter();
}
} }
~DiskWriterEntry() { ~DiskWriterEntry() {
if(enabled) {
delete diskWriter; delete diskWriter;
} }
}
}; };
typedef deque<DiskWriterEntry*> DiskWriterEntries; typedef deque<DiskWriterEntry*> DiskWriterEntries;
@ -49,7 +44,7 @@ typedef deque<DiskWriterEntry*> DiskWriterEntries;
class MultiDiskWriter : public DiskWriter { class MultiDiskWriter : public DiskWriter {
private: private:
DiskWriterEntries diskWriterEntries; DiskWriterEntries diskWriterEntries;
int pieceLength;
bool isInRange(const DiskWriterEntry* entry, long long int offset) const; bool isInRange(const DiskWriterEntry* entry, long long int offset) const;
int calculateLength(const DiskWriterEntry* entry, long long int fileOffset, int rem) const; int calculateLength(const DiskWriterEntry* entry, long long int fileOffset, int rem) const;
void clearEntries(); void clearEntries();
@ -59,10 +54,10 @@ private:
#endif // ENABLE_SHA1DIGEST #endif // ENABLE_SHA1DIGEST
public: public:
MultiDiskWriter(); MultiDiskWriter(int pieceLength);
virtual ~MultiDiskWriter(); virtual ~MultiDiskWriter();
void setMultiFileEntries(const MultiFileEntries& multiFileEntries, int pieceLength); void setFileEntries(const FileEntries& fileEntries);
virtual void openFile(const string& filename); virtual void openFile(const string& filename);
virtual void initAndOpenFile(string filename); virtual void initAndOpenFile(string filename);

View File

@ -216,14 +216,14 @@ void PeerConnection::sendPiece(int index, int begin, int length) const {
int iteration = length/BUF_SIZE; int iteration = length/BUF_SIZE;
long long int pieceOffset = ((long long int)index*torrentMan->pieceLength)+begin; long long int pieceOffset = ((long long int)index*torrentMan->pieceLength)+begin;
for(int i = 0; i < iteration; i++) { for(int i = 0; i < iteration; i++) {
if(torrentMan->diskWriter->readData(buf, BUF_SIZE, pieceOffset+i*BUF_SIZE) < BUF_SIZE) { if(torrentMan->diskAdaptor->readData(buf, BUF_SIZE, pieceOffset+i*BUF_SIZE) < BUF_SIZE) {
throw new DlAbortEx("piece reading failed."); throw new DlAbortEx("piece reading failed.");
} }
socket->writeData(buf, BUF_SIZE); socket->writeData(buf, BUF_SIZE);
} }
int rem = length%BUF_SIZE; int rem = length%BUF_SIZE;
if(rem > 0) { if(rem > 0) {
if(torrentMan->diskWriter->readData(buf, rem, pieceOffset+iteration*BUF_SIZE) < rem) { if(torrentMan->diskAdaptor->readData(buf, rem, pieceOffset+iteration*BUF_SIZE) < rem) {
throw new DlAbortEx("piece reading failed."); throw new DlAbortEx("piece reading failed.");
} }
socket->writeData(buf, rem); socket->writeData(buf, rem);
@ -259,7 +259,7 @@ int PeerConnection::sendPieceData(long long int offset, int length) const {
if(!isWritable) { if(!isWritable) {
return writtenLength; return writtenLength;
} }
if(torrentMan->diskWriter->readData(buf, BUF_SIZE, offset+i*BUF_SIZE) < BUF_SIZE) { if(torrentMan->diskAdaptor->readData(buf, BUF_SIZE, offset+i*BUF_SIZE) < BUF_SIZE) {
throw new DlAbortEx("piece reading failed."); throw new DlAbortEx("piece reading failed.");
} }
socket->writeData(buf, BUF_SIZE); socket->writeData(buf, BUF_SIZE);
@ -268,7 +268,7 @@ int PeerConnection::sendPieceData(long long int offset, int length) const {
if(socket->isWritable(0)) { if(socket->isWritable(0)) {
int rem = length%BUF_SIZE; int rem = length%BUF_SIZE;
if(rem > 0) { if(rem > 0) {
if(torrentMan->diskWriter->readData(buf, rem, offset+iteration*BUF_SIZE) < rem) { if(torrentMan->diskAdaptor->readData(buf, rem, offset+iteration*BUF_SIZE) < rem) {
throw new DlAbortEx("piece reading failed."); throw new DlAbortEx("piece reading failed.");
} }
socket->writeData(buf, rem); socket->writeData(buf, rem);

View File

@ -274,7 +274,7 @@ void PeerInteractionCommand::receiveMessage() {
((long long int)message->getIndex())*e->torrentMan->pieceLength+message->getBegin(); ((long long int)message->getIndex())*e->torrentMan->pieceLength+message->getBegin();
e->logger->debug("CUID#%d - write block length = %d, offset=%lld", e->logger->debug("CUID#%d - write block length = %d, offset=%lld",
cuid, message->getBlockLength(), offset); cuid, message->getBlockLength(), offset);
e->torrentMan->diskWriter->writeData(message->getBlock(), e->torrentMan->diskAdaptor->writeData(message->getBlock(),
message->getBlockLength(), message->getBlockLength(),
offset); offset);
piece.completeBlock(slot.getBlockIndex()); piece.completeBlock(slot.getBlockIndex());
@ -444,7 +444,7 @@ void PeerInteractionCommand::beforeSocketCheck() {
bool PeerInteractionCommand::checkPieceHash(const Piece& piece) { bool PeerInteractionCommand::checkPieceHash(const Piece& piece) {
long long int offset = ((long long int)piece.getIndex())*e->torrentMan->pieceLength; long long int offset = ((long long int)piece.getIndex())*e->torrentMan->pieceLength;
return e->torrentMan->diskWriter->sha1Sum(offset, piece.getLength()) == return e->torrentMan->diskAdaptor->sha1Sum(offset, piece.getLength()) ==
e->torrentMan->getPieceHash(piece.getIndex()); e->torrentMan->getPieceHash(piece.getIndex());
} }
@ -454,11 +454,11 @@ void PeerInteractionCommand::erasePieceOnDisk(const Piece& piece) {
memset(buf, 0, BUFSIZE); memset(buf, 0, BUFSIZE);
long long int offset = ((long long int)piece.getIndex())*e->torrentMan->pieceLength; long long int offset = ((long long int)piece.getIndex())*e->torrentMan->pieceLength;
for(int i = 0; i < piece.getLength()/BUFSIZE; i++) { for(int i = 0; i < piece.getLength()/BUFSIZE; i++) {
e->torrentMan->diskWriter->writeData(buf, BUFSIZE, offset); e->torrentMan->diskAdaptor->writeData(buf, BUFSIZE, offset);
offset += BUFSIZE; offset += BUFSIZE;
} }
int r = piece.getLength()%BUFSIZE; int r = piece.getLength()%BUFSIZE;
if(r > 0) { if(r > 0) {
e->torrentMan->diskWriter->writeData(buf, r, offset); e->torrentMan->diskAdaptor->writeData(buf, r, offset);
} }
} }

View File

@ -35,7 +35,6 @@ PreAllocationDiskWriter::~PreAllocationDiskWriter() {}
void PreAllocationDiskWriter::initAndOpenFile(string filename) { void PreAllocationDiskWriter::initAndOpenFile(string filename) {
createFile(filename); createFile(filename);
int bufSize = 4096; int bufSize = 4096;
char buf[4096]; char buf[4096];

View File

@ -26,8 +26,9 @@ TorrentConsoleDownloadEngine::TorrentConsoleDownloadEngine() {}
TorrentConsoleDownloadEngine::~TorrentConsoleDownloadEngine() {} TorrentConsoleDownloadEngine::~TorrentConsoleDownloadEngine() {}
void TorrentConsoleDownloadEngine::onPartialDownloadingCompletes() { void TorrentConsoleDownloadEngine::onSelectiveDownloadingCompletes() {
printf("Download of specified files has completed. Continue normal download operation.\n"); printf("\nDownload of selected files has completed.\n");
fflush(stdout);
} }
void TorrentConsoleDownloadEngine::printStatistics() { void TorrentConsoleDownloadEngine::printStatistics() {
@ -68,9 +69,9 @@ void TorrentConsoleDownloadEngine::initStatistics() {
sessionDownloadLength = 0; sessionDownloadLength = 0;
downloadLength = 0; downloadLength = 0;
totalLength = 0; totalLength = 0;
if(torrentMan->isPartialDownloadingMode()) { if(torrentMan->isSelectiveDownloadingMode()) {
partialDownloadLengthDiff = torrentMan->getDownloadLength()-torrentMan->getCompletedLength(); selectedDownloadLengthDiff = torrentMan->getDownloadLength()-torrentMan->getCompletedLength();
partialTotalLength = torrentMan->getPartialTotalLength(); selectedTotalLength = torrentMan->getSelectedTotalLength();
} }
} }
@ -95,9 +96,9 @@ void TorrentConsoleDownloadEngine::calculateStatistics() {
torrentMan->resetDeltaDownloadLength(); torrentMan->resetDeltaDownloadLength();
torrentMan->resetDeltaUploadLength(); torrentMan->resetDeltaUploadLength();
if(torrentMan->isPartialDownloadingMode()) { if(torrentMan->isSelectiveDownloadingMode()) {
downloadLength = torrentMan->getDownloadLength()-partialDownloadLengthDiff; downloadLength = torrentMan->getDownloadLength()-selectedDownloadLengthDiff;
totalLength = partialTotalLength; totalLength = selectedTotalLength;
} else { } else {
downloadLength = torrentMan->getDownloadLength(); downloadLength = torrentMan->getDownloadLength();
totalLength = torrentMan->getTotalLength(); totalLength = torrentMan->getTotalLength();

View File

@ -26,11 +26,6 @@
class TorrentConsoleDownloadEngine : public TorrentDownloadEngine { class TorrentConsoleDownloadEngine : public TorrentDownloadEngine {
private: private:
/*
struct timeval cp;
long long int sessionDownloadSize;
long long int sessionUploadSize;
*/
struct timeval cp[2]; struct timeval cp[2];
long long int sessionDownloadLengthArray[2]; long long int sessionDownloadLengthArray[2];
long long int sessionUploadLengthArray[2]; long long int sessionUploadLengthArray[2];
@ -39,11 +34,15 @@ private:
int downloadSpeed; int downloadSpeed;
int uploadSpeed; int uploadSpeed;
int lastElapsed; int lastElapsed;
long long int partialDownloadLengthDiff; long long int selectedDownloadLengthDiff;
long long int partialTotalLength; long long int selectedTotalLength;
// The time when startup
struct timeval startup; struct timeval startup;
// The number of bytes downloaded since startup
long long int sessionDownloadLength; long long int sessionDownloadLength;
// The average speed(bytes per second) since startup
int avgSpeed; int avgSpeed;
// The estimated remaining time to complete the download.
int eta; int eta;
long long int downloadLength; long long int downloadLength;
long long int totalLength; long long int totalLength;
@ -53,7 +52,7 @@ private:
protected: protected:
void initStatistics(); void initStatistics();
void calculateStatistics(); void calculateStatistics();
void onPartialDownloadingCompletes(); void onSelectiveDownloadingCompletes();
public: public:
TorrentConsoleDownloadEngine(); TorrentConsoleDownloadEngine();
~TorrentConsoleDownloadEngine(); ~TorrentConsoleDownloadEngine();

View File

@ -22,7 +22,7 @@
#include "TorrentDownloadEngine.h" #include "TorrentDownloadEngine.h"
void TorrentDownloadEngine::onEndOfRun() { void TorrentDownloadEngine::onEndOfRun() {
torrentMan->diskWriter->closeFile(); torrentMan->diskAdaptor->closeFile();
if(filenameFixed && torrentMan->downloadComplete()) { if(filenameFixed && torrentMan->downloadComplete()) {
torrentMan->remove(); torrentMan->remove();
} else { } else {
@ -32,8 +32,8 @@ void TorrentDownloadEngine::onEndOfRun() {
void TorrentDownloadEngine::afterEachIteration() { void TorrentDownloadEngine::afterEachIteration() {
if(!filenameFixed && torrentMan->downloadComplete()) { if(!filenameFixed && torrentMan->downloadComplete()) {
if(torrentMan->isPartialDownloadingMode()) { if(torrentMan->isSelectiveDownloadingMode()) {
onPartialDownloadingCompletes(); onSelectiveDownloadingCompletes();
} }
torrentMan->onDownloadComplete(); torrentMan->onDownloadComplete();
if(torrentMan->downloadComplete()) { if(torrentMan->downloadComplete()) {

View File

@ -31,7 +31,7 @@ private:
protected: protected:
void onEndOfRun(); void onEndOfRun();
void afterEachIteration(); void afterEachIteration();
virtual void onPartialDownloadingCompletes() = 0; virtual void onSelectiveDownloadingCompletes() = 0;
public: public:
TorrentDownloadEngine():filenameFixed(false) {} TorrentDownloadEngine():filenameFixed(false) {}
virtual ~TorrentDownloadEngine() {} virtual ~TorrentDownloadEngine() {}

View File

@ -32,6 +32,9 @@
#include "DefaultDiskWriter.h" #include "DefaultDiskWriter.h"
#include "MultiDiskWriter.h" #include "MultiDiskWriter.h"
#include "prefs.h" #include "prefs.h"
#include "CopyDiskAdaptor.h"
#include "DirectDiskAdaptor.h"
#include "MultiDiskAdaptor.h"
#include <errno.h> #include <errno.h>
#include <libgen.h> #include <libgen.h>
#include <string.h> #include <string.h>
@ -42,25 +45,21 @@ TorrentMan::TorrentMan():bitfield(NULL),
preDownloadLength(0), preUploadLength(0), preDownloadLength(0), preUploadLength(0),
deltaDownloadLength(0), deltaUploadLength(0), deltaDownloadLength(0), deltaUploadLength(0),
storeDir("."), storeDir("."),
multiFileTopDir(NULL),
setupComplete(false), setupComplete(false),
interval(DEFAULT_ANNOUNCE_INTERVAL), interval(DEFAULT_ANNOUNCE_INTERVAL),
minInterval(DEFAULT_ANNOUNCE_MIN_INTERVAL), minInterval(DEFAULT_ANNOUNCE_MIN_INTERVAL),
complete(0), incomplete(0), complete(0), incomplete(0),
connections(0), diskWriter(NULL) {} connections(0), diskAdaptor(NULL) {}
TorrentMan::~TorrentMan() { TorrentMan::~TorrentMan() {
if(bitfield != NULL) { if(bitfield != NULL) {
delete bitfield; delete bitfield;
} }
if(multiFileTopDir != NULL) {
delete multiFileTopDir;
}
for(Peers::iterator itr = peers.begin(); itr != peers.end(); itr++) { for(Peers::iterator itr = peers.begin(); itr != peers.end(); itr++) {
delete *itr; delete *itr;
} }
if(diskWriter != NULL) { if(diskAdaptor != NULL) {
delete diskWriter; delete diskAdaptor;
} }
} }
@ -298,7 +297,7 @@ bool TorrentMan::downloadComplete() const {
return bitfield->isAllBitSet(); return bitfield->isAllBitSet();
} }
void TorrentMan::readFileEntry(const Dictionary* infoDic, const string& defaultName) { void TorrentMan::readFileEntry(FileEntries& fileEntries, Directory** pTopDir, const Dictionary* infoDic, const string& defaultName) {
Data* topName = (Data*)infoDic->get("name"); Data* topName = (Data*)infoDic->get("name");
if(topName != NULL) { if(topName != NULL) {
name = topName->toString(); name = topName->toString();
@ -313,12 +312,14 @@ void TorrentMan::readFileEntry(const Dictionary* infoDic, const string& defaultN
setFileMode(SINGLE); setFileMode(SINGLE);
Data* length = (Data*)infoDic->get("length"); Data* length = (Data*)infoDic->get("length");
totalLength = length->toLLInt(); totalLength = length->toLLInt();
FileEntry fileEntry(name, totalLength, 0);
fileEntries.push_back(fileEntry);
} else { } else {
long long int length = 0; long long int length = 0;
long long int offset = 0; long long int offset = 0;
// multi-file mode // multi-file mode
setFileMode(MULTI); setFileMode(MULTI);
multiFileTopDir = new Directory(name); *pTopDir = new Directory(name);
const MetaList& metaList = files->getList(); const MetaList& metaList = files->getList();
for(MetaList::const_iterator itr = metaList.begin(); itr != metaList.end(); for(MetaList::const_iterator itr = metaList.begin(); itr != metaList.end();
itr++) { itr++) {
@ -327,7 +328,7 @@ void TorrentMan::readFileEntry(const Dictionary* infoDic, const string& defaultN
length += lengthData->toLLInt(); length += lengthData->toLLInt();
List* path = (List*)fileDic->get("path"); List* path = (List*)fileDic->get("path");
const MetaList& paths = path->getList(); const MetaList& paths = path->getList();
Directory* parentDir = multiFileTopDir; Directory* parentDir = *pTopDir;
string filePath = name; string filePath = name;
for(int i = 0; i < (int)paths.size()-1; i++) { for(int i = 0; i < (int)paths.size()-1; i++) {
Data* subpath = (Data*)paths.at(i); Data* subpath = (Data*)paths.at(i);
@ -339,14 +340,14 @@ void TorrentMan::readFileEntry(const Dictionary* infoDic, const string& defaultN
Data* lastpath = (Data*)paths.back(); Data* lastpath = (Data*)paths.back();
filePath.append("/").append(lastpath->toString()); filePath.append("/").append(lastpath->toString());
FileEntry fileEntry(filePath, lengthData->toLLInt(), offset); FileEntry fileEntry(filePath, lengthData->toLLInt(), offset);
multiFileEntries.push_back(fileEntry); fileEntries.push_back(fileEntry);
offset += fileEntry.length; offset += fileEntry.length;
} }
totalLength = length; totalLength = length;
} }
} }
void TorrentMan::setup(string metaInfoFile) { void TorrentMan::setup(string metaInfoFile, const Strings& targetFilePaths) {
peerId = "-A2****-"; peerId = "-A2****-";
for(int i = 0; i < 12; i++) { for(int i = 0; i < 12; i++) {
peerId += Util::itos((int)(((double)10)*random()/(RAND_MAX+1.0))); peerId += Util::itos((int)(((double)10)*random()/(RAND_MAX+1.0)));
@ -363,7 +364,9 @@ void TorrentMan::setup(string metaInfoFile) {
v.getHash(md, len); v.getHash(md, len);
setInfoHash(md); setInfoHash(md);
readFileEntry(infoDic, metaInfoFile); FileEntries fileEntries;
Directory* topDir = NULL;
readFileEntry(fileEntries, &topDir, infoDic, metaInfoFile);
announce = ((Data*)topDic->get("announce"))->toString(); announce = ((Data*)topDic->get("announce"))->toString();
pieceLength = ((Data*)infoDic->get("piece length"))->toInt(); pieceLength = ((Data*)infoDic->get("piece length"))->toInt();
@ -380,45 +383,54 @@ void TorrentMan::setup(string metaInfoFile) {
initBitfield(); initBitfield();
delete topDic; delete topDic;
}
void TorrentMan::setupDiskWriter() {
if(option->get(PREF_DIRECT_FILE_MAPPING) == V_TRUE) { if(option->get(PREF_DIRECT_FILE_MAPPING) == V_TRUE) {
if(segmentFileExists()) {
load();
}
if(fileMode == SINGLE) { if(fileMode == SINGLE) {
diskWriter = new DefaultDiskWriter(); diskAdaptor = new DirectDiskAdaptor(new DefaultDiskWriter(totalLength));
} else { } else {
diskWriter = new MultiDiskWriter(); diskAdaptor = new MultiDiskAdaptor(new MultiDiskWriter(pieceLength));
((MultiDiskWriter*)diskWriter)->setMultiFileEntries(multiFileEntries, pieceLength);
multiFileTopDir->createDir(storeDir, true);
} }
diskWriter->openFile(getFilePath());
} else { } else {
if(option->get(PREF_NO_PREALLOCATION) == V_TRUE) { diskAdaptor = new CopyDiskAdaptor(new DefaultDiskWriter(totalLength));
diskWriter = new DefaultDiskWriter(); ((CopyDiskAdaptor*)diskAdaptor)->setTempFilename(name+".a2tmp");
} else {
diskWriter = new PreAllocationDiskWriter(totalLength);
} }
diskAdaptor->setStoreDir(storeDir);
diskAdaptor->setTopDir(topDir);
diskAdaptor->setFileEntries(fileEntries);
setFileFilter(targetFilePaths);
if(segmentFileExists()) { if(segmentFileExists()) {
load(); load();
diskWriter->openExistingFile(getTempFilePath()); diskAdaptor->openExistingFile();
} else { } else {
diskWriter->initAndOpenFile(getTempFilePath()); diskAdaptor->initAndOpenFile();
}
} }
setupComplete = true; setupComplete = true;
} }
const MultiFileEntries& TorrentMan::getMultiFileEntries() const { void TorrentMan::setFileFilter(const Strings& filePaths) {
return multiFileEntries; if(fileMode != MULTI || filePaths.empty()) {
return;
}
diskAdaptor->removeAllDownloadEntry();
for(Strings::const_iterator pitr = filePaths.begin();
pitr != filePaths.end(); pitr++) {
if(!diskAdaptor->addDownloadEntry(*pitr)) {
throw new DlAbortEx("no such file entry <%s>", (*pitr).c_str());
}
FileEntry fileEntry = diskAdaptor->getFileEntryFromPath(*pitr);
bitfield->addFilter(fileEntry.offset, fileEntry.length);
}
bitfield->enableFilter();
} }
void TorrentMan::readFileEntryFromMetaInfoFile(const string& metaInfoFile) { FileEntries TorrentMan::readFileEntryFromMetaInfoFile(const string& metaInfoFile) {
Dictionary* topDic = (Dictionary*)MetaFileUtil::parseMetaFile(metaInfoFile); Dictionary* topDic = (Dictionary*)MetaFileUtil::parseMetaFile(metaInfoFile);
const Dictionary* infoDic = (const Dictionary*)topDic->get("info"); const Dictionary* infoDic = (const Dictionary*)topDic->get("info");
readFileEntry(infoDic, metaInfoFile); FileEntries fileEntries;
Directory* topDir;
readFileEntry(fileEntries, &topDir, infoDic, metaInfoFile);
delete topDir;
return fileEntries;
} }
string TorrentMan::getName() const { string TorrentMan::getName() const {
@ -433,28 +445,8 @@ string TorrentMan::getPieceHash(int index) const {
return pieceHashes.at(index); return pieceHashes.at(index);
} }
string TorrentMan::getFilePath() const {
if(option->get(PREF_DIRECT_FILE_MAPPING) == V_TRUE && fileMode == MULTI) {
return storeDir;
} else {
return storeDir+"/"+name;
}
}
string TorrentMan::getTempFilePath() const {
if(option->get(PREF_DIRECT_FILE_MAPPING) == V_TRUE) {
return getFilePath();
} else {
return getFilePath()+".a2tmp";
}
}
string TorrentMan::getSegmentFilePath() const { string TorrentMan::getSegmentFilePath() const {
if(option->get(PREF_DIRECT_FILE_MAPPING) == V_TRUE && fileMode == MULTI) {
return storeDir+"/"+name+".aria2"; return storeDir+"/"+name+".aria2";
} else {
return getFilePath()+".aria2";
}
} }
bool TorrentMan::segmentFileExists() const { bool TorrentMan::segmentFileExists() const {
@ -546,117 +538,27 @@ void TorrentMan::remove() const {
} }
} }
void TorrentMan::fixFilename() { bool TorrentMan::isSelectiveDownloadingMode() const {
if(option->get(PREF_DIRECT_FILE_MAPPING) == V_TRUE) {
// nothing to do here
} else {
if(fileMode == SINGLE) {
copySingleFile();
} else {
splitMultiFile();
}
}
}
void TorrentMan::copySingleFile() const {
logger->info("writing file %s", getFilePath().c_str());
Util::fileCopy(getFilePath(), getTempFilePath());
}
void TorrentMan::splitMultiFile() {
logger->info("creating directories");
multiFileTopDir->createDir(storeDir, true);
long long int offset = 0;
for(MultiFileEntries::iterator itr = multiFileEntries.begin();
itr != multiFileEntries.end(); itr++) {
if(!itr->extracted && itr->requested) {
string dest = storeDir+"/"+itr->path;
logger->info("writing file %s", dest.c_str());
Util::rangedFileCopy(dest, getTempFilePath(), offset, itr->length);
itr->extracted = true;
}
offset += itr->length;
}
}
void TorrentMan::deleteTempFile() const {
if(option->get(PREF_DIRECT_FILE_MAPPING) == V_TRUE) {
// nothing to do here
} else {
unlink(getTempFilePath().c_str());
}
}
// bool TorrentMan::unextractedFileEntryExists() const {
// if(fileMode == SINGLE) {
// return false;
// }
// for(MultiFileEntries::const_iterator itr = multiFileEntries.begin();
// itr != multiFileEntries.end(); itr++) {
// if(!itr->extracted) {
// return true;
// }
// }
// }
void TorrentMan::setFileEntriesToDownload(const Strings& filePaths) {
if(fileMode != MULTI) {
throw new DlAbortEx("only multi-mode supports partial downloading mode.");
}
// clear all requested flags in multiFileEntries.
setAllMultiFileRequestedState(false);
for(Strings::const_iterator pitr = filePaths.begin();
pitr != filePaths.end(); pitr++) {
bool found = false;
for(MultiFileEntries::iterator itr = multiFileEntries.begin();
itr != multiFileEntries.end(); itr++) {
if(*pitr == itr->path) {
itr->requested = true;
found = true;
bitfield->addFilter(itr->offset, itr->length);
break;
}
}
if(!found) {
throw new DlAbortEx("no such file entry <%s>", (*pitr).c_str());
}
}
bitfield->enableFilter();
}
bool TorrentMan::isPartialDownloadingMode() const {
return bitfield->isFilterEnabled(); return bitfield->isFilterEnabled();
} }
void TorrentMan::setAllMultiFileRequestedState(bool state) { void TorrentMan::finishSelectiveDownloadingMode() {
for(MultiFileEntries::iterator itr = multiFileEntries.begin();
itr != multiFileEntries.end(); itr++) {
itr->requested = state;
}
}
void TorrentMan::finishPartialDownloadingMode() {
bitfield->clearFilter(); bitfield->clearFilter();
setAllMultiFileRequestedState(true); diskAdaptor->addAllDownloadEntry();
if(option->get(PREF_DIRECT_FILE_MAPPING) == V_TRUE && fileMode == MULTI) {
((MultiDiskWriter*)diskWriter)->setMultiFileEntries(multiFileEntries, pieceLength);
}
} }
long long int TorrentMan::getCompletedLength() const { long long int TorrentMan::getCompletedLength() const {
return bitfield->getCompletedLength(); return bitfield->getCompletedLength();
} }
long long int TorrentMan::getPartialTotalLength() const { long long int TorrentMan::getSelectedTotalLength() const {
return bitfield->getFilteredTotalLength(); return bitfield->getFilteredTotalLength();
} }
void TorrentMan::onDownloadComplete() { void TorrentMan::onDownloadComplete() {
diskWriter->closeFile();
save(); save();
fixFilename(); diskAdaptor->onDownloadComplete();
if(isPartialDownloadingMode()) { if(isSelectiveDownloadingMode()) {
finishPartialDownloadingMode(); finishSelectiveDownloadingMode();
} }
diskWriter->openFile(getTempFilePath());
} }

View File

@ -28,9 +28,10 @@
#include "BitfieldMan.h" #include "BitfieldMan.h"
#include "DiskWriter.h" #include "DiskWriter.h"
#include "Piece.h" #include "Piece.h"
#include "Directory.h"
#include "Dictionary.h" #include "Dictionary.h"
#include "Option.h" #include "Option.h"
#include "FileEntry.h"
#include "DiskAdaptor.h"
#include <deque> #include <deque>
#include <map> #include <map>
#include <string> #include <string>
@ -46,22 +47,8 @@ using namespace std;
#define END_GAME_PIECE_NUM 20 #define END_GAME_PIECE_NUM 20
#define MAX_PEER_ERROR 5 #define MAX_PEER_ERROR 5
class FileEntry {
public:
string path;
long long int length;
long long int offset;
bool extracted;
bool requested;
FileEntry(string path, long long int length, long long int offset):
path(path), length(length), offset(offset),
extracted(false), requested(true) {}
~FileEntry() {}
};
typedef deque<Peer*> Peers; typedef deque<Peer*> Peers;
typedef multimap<int, int> Haves; typedef multimap<int, int> Haves;
typedef deque<FileEntry> MultiFileEntries;
typedef deque<Piece> UsedPieces; typedef deque<Piece> UsedPieces;
typedef deque<int> PieceIndexes; typedef deque<int> PieceIndexes;
@ -85,8 +72,6 @@ private:
int port; int port;
Haves haves; Haves haves;
UsedPieces usedPieces; UsedPieces usedPieces;
Directory* multiFileTopDir;
MultiFileEntries multiFileEntries;
bool setupComplete; bool setupComplete;
FILE* openSegFile(string segFilename, string mode) const; FILE* openSegFile(string segFilename, string mode) const;
@ -97,7 +82,8 @@ private:
void deleteUsedPiece(const Piece& piece); void deleteUsedPiece(const Piece& piece);
int deleteUsedPiecesByFillRate(int fillRate, int toDelete); int deleteUsedPiecesByFillRate(int fillRate, int toDelete);
void reduceUsedPieces(int max); void reduceUsedPieces(int max);
void readFileEntry(const Dictionary* infoDic, const string& defaultName); void readFileEntry(FileEntries& fileEntries, Directory** pTopDir, const Dictionary* infoDic, const string& defaultName);
void setFileFilter(const Strings& filePaths);
public: public:
int pieceLength; int pieceLength;
int pieces; int pieces;
@ -115,7 +101,7 @@ public:
~TorrentMan(); ~TorrentMan();
const Logger* logger; const Logger* logger;
DiskWriter* diskWriter; DiskAdaptor* diskAdaptor;
const Option* option; const Option* option;
int getNewCuid() { return ++cuidCounter; } int getNewCuid() { return ++cuidCounter; }
@ -150,8 +136,7 @@ public:
return infoHash; return infoHash;
} }
void setup(string metaInfoFile); void setup(string metaInfoFile, const Strings& targetFilePaths);
void setupDiskWriter();
string getPieceHash(int index) const; string getPieceHash(int index) const;
@ -213,8 +198,6 @@ public:
string getStoreDir() const { return storeDir; } string getStoreDir() const { return storeDir; }
void setStoreDir(string dir) { storeDir = dir; } void setStoreDir(string dir) { storeDir = dir; }
string getFilePath() const;
string getTempFilePath() const;
string getSegmentFilePath() const; string getSegmentFilePath() const;
bool segmentFileExists() const; bool segmentFileExists() const;
@ -233,20 +216,14 @@ public:
int countUsedPiece() const { return usedPieces.size(); } int countUsedPiece() const { return usedPieces.size(); }
int countAdvertisedPiece() const { return haves.size(); } int countAdvertisedPiece() const { return haves.size(); }
void readFileEntryFromMetaInfoFile(const string& metaInfoFile); FileEntries readFileEntryFromMetaInfoFile(const string& metaInfoFile);
const MultiFileEntries& getMultiFileEntries() const;
string getName() const; string getName() const;
//bool unextractedFileEntryExists() const; void finishSelectiveDownloadingMode();
bool isSelectiveDownloadingMode() const;
void setAllMultiFileRequestedState(bool state);
void finishPartialDownloadingMode();
bool isPartialDownloadingMode() const;
void setFileEntriesToDownload(const Strings& filePaths);
long long int getCompletedLength() const; long long int getCompletedLength() const;
long long int getPartialTotalLength() const; long long int getSelectedTotalLength() const;
void onDownloadComplete(); void onDownloadComplete();

View File

@ -63,6 +63,10 @@ void printDownloadCompeleteMessage(string filename) {
printf(_("\nThe download was complete. <%s>\n"), filename.c_str()); printf(_("\nThe download was complete. <%s>\n"), filename.c_str());
} }
void printDownloadCompeleteMessage() {
printf("\nThe download was complete.\n");
}
void printDownloadAbortMessage() { void printDownloadAbortMessage() {
printf(_("\nThe download was not complete because of errors. Check the log.\n")); printf(_("\nThe download was not complete because of errors. Check the log.\n"));
} }
@ -85,13 +89,13 @@ void handler(int signal) {
void torrentHandler(int signal) { void torrentHandler(int signal) {
cout << _("\nSIGINT signal received.") << endl; cout << _("\nSIGINT signal received.") << endl;
if(te->torrentMan->diskWriter != NULL) { if(te->torrentMan->diskAdaptor != NULL) {
te->torrentMan->diskWriter->closeFile(); te->torrentMan->diskAdaptor->closeFile();
} }
if(te->torrentMan->downloadComplete() && te->isFilenameFixed()) { if(te->torrentMan->downloadComplete() && te->isFilenameFixed()) {
te->torrentMan->remove(); te->torrentMan->remove();
te->torrentMan->deleteTempFile(); //te->torrentMan->deleteTempFile();
printDownloadCompeleteMessage(te->torrentMan->getFilePath()); printDownloadCompeleteMessage();
} else { } else {
te->torrentMan->save(); te->torrentMan->save();
} }
@ -283,7 +287,7 @@ int main(int argc, char* argv[]) {
#ifdef ENABLE_BITTORRENT #ifdef ENABLE_BITTORRENT
{ "torrent-file", required_argument, &lopt, 15 }, { "torrent-file", required_argument, &lopt, 15 },
{ "follow-torrent", required_argument, &lopt, 16 }, { "follow-torrent", required_argument, &lopt, 16 },
{ "torrent-show-files", no_argument, &lopt, 17 }, { "show-files", no_argument, &lopt, 17 },
{ "no-preallocation", no_argument, &lopt, 18 }, { "no-preallocation", no_argument, &lopt, 18 },
{ "direct-file-mapping", required_argument, &lopt, 19 }, { "direct-file-mapping", required_argument, &lopt, 19 },
#endif // ENABLE_BITTORRENT #endif // ENABLE_BITTORRENT
@ -414,7 +418,7 @@ int main(int argc, char* argv[]) {
} }
break; break;
case 17: case 17:
op->put(PREF_TORRENT_SHOW_FILES, V_TRUE); op->put(PREF_SHOW_FILES, V_TRUE);
break; break;
case 18: case 18:
op->put(PREF_NO_PREALLOCATION, V_TRUE); op->put(PREF_NO_PREALLOCATION, V_TRUE);
@ -608,32 +612,21 @@ int main(int argc, char* argv[]) {
te->torrentMan->option = op; te->torrentMan->option = op;
string targetTorrentFile = torrentFile.empty() ? string targetTorrentFile = torrentFile.empty() ?
downloadedTorrentFile : torrentFile; downloadedTorrentFile : torrentFile;
if(op->get(PREF_TORRENT_SHOW_FILES) == V_TRUE) { if(op->get(PREF_SHOW_FILES) == V_TRUE) {
te->torrentMan->readFileEntryFromMetaInfoFile(targetTorrentFile); FileEntries fileEntries = te->torrentMan->readFileEntryFromMetaInfoFile(targetTorrentFile);
cout << "Files:" << endl; cout << "Files:" << endl;
switch(te->torrentMan->getFileMode()) { for(FileEntries::const_iterator itr = fileEntries.begin();
case TorrentMan::SINGLE: itr != fileEntries.end(); itr++) {
printf("%s %s Bytes\n", te->torrentMan->getName().c_str(),
Util::llitos(te->torrentMan->getTotalLength(), true).c_str());
break;
case TorrentMan::MULTI: {
const MultiFileEntries& entries = te->torrentMan->getMultiFileEntries();
for(MultiFileEntries::const_iterator itr = entries.begin();
itr != entries.end(); itr++) {
printf("%s %s Bytes\n", itr->path.c_str(), printf("%s %s Bytes\n", itr->path.c_str(),
Util::llitos(itr->length, true).c_str()); Util::llitos(itr->length, true).c_str());
} }
break;
}
}
exit(0); exit(0);
} else { } else {
te->torrentMan->setup(targetTorrentFile); Strings targetFiles;
if(!torrentFile.empty() && !args.empty() && if(!torrentFile.empty() && !args.empty()) {
te->torrentMan->getFileMode() == TorrentMan::MULTI) { targetFiles = args;
te->torrentMan->setFileEntriesToDownload(args);
} }
te->torrentMan->setupDiskWriter(); te->torrentMan->setup(targetTorrentFile, targetFiles);
} }
PeerListenCommand* listenCommand = PeerListenCommand* listenCommand =
new PeerListenCommand(te->torrentMan->getNewCuid(), te); new PeerListenCommand(te->torrentMan->getNewCuid(), te);
@ -652,7 +645,7 @@ int main(int argc, char* argv[]) {
te->run(); te->run();
if(te->torrentMan->downloadComplete()) { if(te->torrentMan->downloadComplete()) {
printDownloadCompeleteMessage(te->torrentMan->getFilePath()); printDownloadCompeleteMessage();
} else { } else {
printDownloadAbortMessage(); printDownloadAbortMessage();
} }

View File

@ -89,7 +89,7 @@
// values: 1*digit // values: 1*digit
#define PREF_PEER_CONNECTION_TIMEOUT "peer_connection_timeout" #define PREF_PEER_CONNECTION_TIMEOUT "peer_connection_timeout"
// values: true | false // values: true | false
#define PREF_TORRENT_SHOW_FILES "torrent_show_files" #define PREF_SHOW_FILES "show_files"
// values: true | false // values: true | false
#define PREF_NO_PREALLOCATION "no_preallocation" #define PREF_NO_PREALLOCATION "no_preallocation"
// values: true | false // values: true | false

View File

@ -25,14 +25,14 @@ public:
CPPUNIT_TEST_SUITE_REGISTRATION( MultiDiskWriterTest ); CPPUNIT_TEST_SUITE_REGISTRATION( MultiDiskWriterTest );
MultiFileEntries createEntries() { FileEntries createEntries() {
FileEntry entry1("file1.txt", 15, 0); FileEntry entry1("file1.txt", 15, 0);
FileEntry entry2("file2.txt", 7, 15); FileEntry entry2("file2.txt", 7, 15);
FileEntry entry3("file3.txt", 3, 22); FileEntry entry3("file3.txt", 3, 22);
unlink("file1.txt"); unlink("file1.txt");
unlink("file2.txt"); unlink("file2.txt");
unlink("file3.txt"); unlink("file3.txt");
MultiFileEntries entries; FileEntries entries;
entries.push_back(entry1); entries.push_back(entry1);
entries.push_back(entry2); entries.push_back(entry2);
entries.push_back(entry3); entries.push_back(entry3);
@ -52,8 +52,8 @@ void readFile(const string& filename, char* buf, int bufLength) {
} }
void MultiDiskWriterTest::testWriteData() { void MultiDiskWriterTest::testWriteData() {
MultiDiskWriter dw; MultiDiskWriter dw(2);
dw.setMultiFileEntries(createEntries(), 2); dw.setFileEntries(createEntries());
dw.openFile("."); dw.openFile(".");
string msg = "12345"; string msg = "12345";
@ -97,12 +97,12 @@ void MultiDiskWriterTest::testReadData() {
FileEntry entry1("file1r.txt", 15, 0); FileEntry entry1("file1r.txt", 15, 0);
FileEntry entry2("file2r.txt", 7, 15); FileEntry entry2("file2r.txt", 7, 15);
FileEntry entry3("file3r.txt", 3, 22); FileEntry entry3("file3r.txt", 3, 22);
MultiFileEntries entries; FileEntries entries;
entries.push_back(entry1); entries.push_back(entry1);
entries.push_back(entry2); entries.push_back(entry2);
entries.push_back(entry3); entries.push_back(entry3);
MultiDiskWriter dw; MultiDiskWriter dw(2);
dw.setMultiFileEntries(entries, 2); dw.setFileEntries(entries);
dw.openFile("."); dw.openFile(".");
char buf[128]; char buf[128];
@ -124,12 +124,12 @@ void MultiDiskWriterTest::testSha1Sum() {
FileEntry entry1("file1r.txt", 15, 0); FileEntry entry1("file1r.txt", 15, 0);
FileEntry entry2("file2r.txt", 7, 15); FileEntry entry2("file2r.txt", 7, 15);
FileEntry entry3("file3r.txt", 3, 22); FileEntry entry3("file3r.txt", 3, 22);
MultiFileEntries entries; FileEntries entries;
entries.push_back(entry1); entries.push_back(entry1);
entries.push_back(entry2); entries.push_back(entry2);
entries.push_back(entry3); entries.push_back(entry3);
MultiDiskWriter dw; MultiDiskWriter dw(2);
dw.setMultiFileEntries(entries, 2); dw.setFileEntries(entries);
dw.openFile("."); dw.openFile(".");
string sha1sum = dw.sha1Sum(0, 25); string sha1sum = dw.sha1Sum(0, 25);