aria2/src/RequestGroup.h

526 lines
14 KiB
C++

/* <!-- copyright */
/*
* aria2 - The high speed download utility
*
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*
* In addition, as a special exception, the copyright holders give
* permission to link the code of portions of this program with the
* OpenSSL library under certain conditions as described in each
* individual source file, and distribute linked combinations
* including the two.
* You must obey the GNU General Public License in all respects
* for all of the code used other than OpenSSL. If you modify
* file(s) with this exception, you may extend this exception to your
* version of the file(s), but you are not obligated to do so. If you
* do not wish to do so, delete this exception statement from your
* version. If you delete this exception statement from all source
* files in the program, then also delete it here.
*/
/* copyright --> */
#ifndef D_REQUEST_GROUP_H
#define D_REQUEST_GROUP_H
#include "common.h"
#include <string>
#include <algorithm>
#include <vector>
#include <memory>
#include <utility>
#include "TransferStat.h"
#include "TimeA2.h"
#include "Request.h"
#include "error_code.h"
#include "MetadataInfo.h"
#include "GroupId.h"
namespace aria2 {
class DownloadEngine;
class SegmentMan;
class Command;
class DownloadCommand;
class DownloadContext;
class PieceStorage;
class BtProgressInfoFile;
class Dependency;
class PreDownloadHandler;
class PostDownloadHandler;
class DiskWriterFactory;
class Option;
class RequestGroup;
class CheckIntegrityEntry;
struct DownloadResult;
class URISelector;
class URIResult;
class RequestGroupMan;
#ifdef ENABLE_BITTORRENT
class BtRuntime;
class PeerStorage;
#endif // ENABLE_BITTORRENT
class RequestGroup {
public:
enum HaltReason { NONE, SHUTDOWN_SIGNAL, USER_REQUEST };
enum State {
// Waiting in the reserved queue
STATE_WAITING,
// Download has begun
STATE_ACTIVE
};
private:
// If this download is a part of another download(for example,
// downloading torrent file described in Metalink file), this field
// has the GID of parent RequestGroup. 0 means this is a parent
// RequestGroup.
a2_gid_t belongsToGID_;
std::shared_ptr<GroupId> gid_;
std::shared_ptr<Option> option_;
// options applied on restart
std::shared_ptr<Option> pendingOption_;
std::shared_ptr<SegmentMan> segmentMan_;
std::shared_ptr<DownloadContext> downloadContext_;
std::shared_ptr<PieceStorage> pieceStorage_;
std::shared_ptr<BtProgressInfoFile> progressInfoFile_;
std::shared_ptr<DiskWriterFactory> diskWriterFactory_;
std::shared_ptr<Dependency> dependency_;
std::unique_ptr<URISelector> uriSelector_;
std::shared_ptr<MetadataInfo> metadataInfo_;
RequestGroupMan* requestGroupMan_;
#ifdef ENABLE_BITTORRENT
BtRuntime* btRuntime_;
PeerStorage* peerStorage_;
#endif // ENABLE_BITTORRENT
// If this download generates another downloads when completed(for
// example, downloads generated by PostDownloadHandler), this field
// has the GID of generated RequestGroups. empty list means there is
// no such RequestGroup.
std::vector<a2_gid_t> followedByGIDs_;
// This is a reverse link against followedByGIDs_. For example, a
// download included in followedByGIDs_ has this download's GID in
// followingGID_.
a2_gid_t followingGID_;
std::vector<const PreDownloadHandler*> preDownloadHandlers_;
std::vector<const PostDownloadHandler*> postDownloadHandlers_;
Time lastModifiedTime_;
// Timeout used for HTTP/FTP downloads.
std::chrono::seconds timeout_;
int state_;
int numConcurrentCommand_;
/**
* This is the number of connections used in streaming protocol(http/ftp)
*/
int numStreamConnection_;
int numStreamCommand_;
int numCommand_;
int fileNotFoundCount_;
int maxDownloadSpeedLimit_;
int maxUploadSpeedLimit_;
int resumeFailureCount_;
HaltReason haltReason_;
error_code::Value lastErrorCode_;
std::string lastErrorMessage_;
bool saveControlFile_;
bool fileAllocationEnabled_;
bool preLocalFileCheckEnabled_;
bool haltRequested_;
bool forceHaltRequested_;
bool pauseRequested_;
// restartRequested_ indicates that this download should be
// restarted. Usually, it is used with pauseRequested_ to stop
// download first.
bool restartRequested_;
// This flag just indicates that the downloaded file is not saved disk but
// just sits in memory.
bool inMemoryDownload_;
bool seedOnly_;
void validateFilename(const std::string& expectedFilename,
const std::string& actualFilename) const;
void initializePreDownloadHandler();
void initializePostDownloadHandler();
// Returns the result code of this RequestGroup. If the download
// finished, then returns error_code::FINISHED. If the
// download didn't finish and error result is available in
// _uriResults, then last result code is returned. Otherwise
// returns error_code::UNKNOWN_ERROR.
std::pair<error_code::Value, std::string> downloadResult() const;
void removeDefunctControlFile(
const std::shared_ptr<BtProgressInfoFile>& progressInfoFile);
public:
RequestGroup(const std::shared_ptr<GroupId>& gid,
const std::shared_ptr<Option>& option);
~RequestGroup();
bool isCheckIntegrityReady();
void tryAutoFileRenaming();
const std::shared_ptr<SegmentMan>& getSegmentMan() const
{
return segmentMan_;
}
std::unique_ptr<CheckIntegrityEntry> createCheckIntegrityEntry();
// Returns first bootstrap commands to initiate a download.
// If this is HTTP/FTP download and file size is unknown, only 1 command
// (usually, HttpInitiateConnection or FtpInitiateConnection) will be created.
void createInitialCommand(std::vector<std::unique_ptr<Command>>& commands,
DownloadEngine* e);
void createNextCommandWithAdj(std::vector<std::unique_ptr<Command>>& commands,
DownloadEngine* e, int numAdj);
void createNextCommand(std::vector<std::unique_ptr<Command>>& commands,
DownloadEngine* e, int numCommand);
void createNextCommand(std::vector<std::unique_ptr<Command>>& commands,
DownloadEngine* e);
bool downloadFinished() const;
bool allDownloadFinished() const;
void closeFile();
std::string getFirstFilePath() const;
int64_t getTotalLength() const;
int64_t getCompletedLength() const;
inline int64_t getPendingLength() const
{
return getTotalLength() - getCompletedLength();
}
/**
* Compares expected filename with specified actualFilename.
* The expected filename refers to FileEntry::getBasename() of the first
* element of DownloadContext::getFileEntries()
*/
void validateFilename(const std::string& actualFilename) const;
void validateTotalLength(int64_t expectedTotalLength,
int64_t actualTotalLength) const;
void validateTotalLength(int64_t actualTotalLength) const;
void setNumConcurrentCommand(int num) { numConcurrentCommand_ = num; }
int getNumConcurrentCommand() const { return numConcurrentCommand_; }
a2_gid_t getGID() const { return gid_->getNumericId(); }
const std::shared_ptr<GroupId>& getGroupId() const { return gid_; }
TransferStat calculateStat() const;
const std::shared_ptr<DownloadContext>& getDownloadContext() const
{
return downloadContext_;
}
// This function also calls
// downloadContext->setOwnerRequestGroup(this).
void
setDownloadContext(const std::shared_ptr<DownloadContext>& downloadContext);
const std::shared_ptr<PieceStorage>& getPieceStorage() const
{
return pieceStorage_;
}
void setPieceStorage(const std::shared_ptr<PieceStorage>& pieceStorage);
void setProgressInfoFile(
const std::shared_ptr<BtProgressInfoFile>& progressInfoFile);
void increaseStreamCommand();
void decreaseStreamCommand();
void increaseStreamConnection();
void decreaseStreamConnection();
int getNumConnection() const;
void increaseNumCommand();
void decreaseNumCommand();
int getNumCommand() const { return numCommand_; }
// TODO is it better to move the following 2 methods to
// SingleFileDownloadContext?
void setDiskWriterFactory(
const std::shared_ptr<DiskWriterFactory>& diskWriterFactory);
const std::shared_ptr<DiskWriterFactory>& getDiskWriterFactory() const
{
return diskWriterFactory_;
}
void setFileAllocationEnabled(bool f) { fileAllocationEnabled_ = f; }
bool isFileAllocationEnabled() const { return fileAllocationEnabled_; }
bool needsFileAllocation() const;
/**
* Setting preLocalFileCheckEnabled_ to false, then skip the check to see
* if a file is already exists and control file exists etc.
* Always open file with DiskAdaptor::initAndOpenFile()
*/
void setPreLocalFileCheckEnabled(bool f) { preLocalFileCheckEnabled_ = f; }
bool isPreLocalFileCheckEnabled() const { return preLocalFileCheckEnabled_; }
void setHaltRequested(bool f, HaltReason = SHUTDOWN_SIGNAL);
void setForceHaltRequested(bool f, HaltReason = SHUTDOWN_SIGNAL);
bool isHaltRequested() const { return haltRequested_; }
bool isForceHaltRequested() const { return forceHaltRequested_; }
void setPauseRequested(bool f);
bool isPauseRequested() const { return pauseRequested_; }
void setRestartRequested(bool f);
bool isRestartRequested() const { return restartRequested_; }
void dependsOn(const std::shared_ptr<Dependency>& dep);
bool isDependencyResolved();
void releaseRuntimeResource(DownloadEngine* e);
void
postDownloadProcessing(std::vector<std::shared_ptr<RequestGroup>>& groups);
void addPostDownloadHandler(const PostDownloadHandler* handler);
void clearPostDownloadHandler();
void preDownloadProcessing();
void addPreDownloadHandler(const PreDownloadHandler* handler);
void clearPreDownloadHandler();
void
processCheckIntegrityEntry(std::vector<std::unique_ptr<Command>>& commands,
std::unique_ptr<CheckIntegrityEntry> entry,
DownloadEngine* e);
// Initializes pieceStorage_ and segmentMan_. We guarantee that
// either both of pieceStorage_ and segmentMan_ are initialized or
// they are not.
void initPieceStorage();
void dropPieceStorage();
bool downloadFinishedByFileLength();
void
loadAndOpenFile(const std::shared_ptr<BtProgressInfoFile>& progressInfoFile);
void shouldCancelDownloadForSafety();
void adjustFilename(const std::shared_ptr<BtProgressInfoFile>& infoFile);
std::shared_ptr<DownloadResult> createDownloadResult() const;
const std::shared_ptr<Option>& getOption() const { return option_; }
void reportDownloadFinished();
void setURISelector(std::unique_ptr<URISelector> uriSelector);
const std::unique_ptr<URISelector>& getURISelector() const
{
return uriSelector_;
}
void applyLastModifiedTimeToLocalFiles();
void updateLastModifiedTime(const Time& time);
void increaseAndValidateFileNotFoundCount();
// Just set inMemoryDownload flag true.
void markInMemoryDownload();
// Returns inMemoryDownload flag.
bool inMemoryDownload() const { return inMemoryDownload_; }
void setTimeout(std::chrono::seconds timeout);
const std::chrono::seconds& getTimeout() const { return timeout_; }
// Returns true if current download speed exceeds
// maxDownloadSpeedLimit_. Always returns false if
// maxDownloadSpeedLimit_ == 0. Otherwise returns false.
bool doesDownloadSpeedExceed();
// Returns true if current upload speed exceeds
// maxUploadSpeedLimit_. Always returns false if
// maxUploadSpeedLimit_ == 0. Otherwise returns false.
bool doesUploadSpeedExceed();
int getMaxDownloadSpeedLimit() const { return maxDownloadSpeedLimit_; }
void setMaxDownloadSpeedLimit(int speed) { maxDownloadSpeedLimit_ = speed; }
int getMaxUploadSpeedLimit() const { return maxUploadSpeedLimit_; }
void setMaxUploadSpeedLimit(int speed) { maxUploadSpeedLimit_ = speed; }
void setLastErrorCode(error_code::Value code, const char* message = "")
{
lastErrorCode_ = code;
lastErrorMessage_ = message;
}
error_code::Value getLastErrorCode() const { return lastErrorCode_; }
void saveControlFile() const;
void removeControlFile() const;
void enableSaveControlFile() { saveControlFile_ = true; }
void disableSaveControlFile() { saveControlFile_ = false; }
template <typename InputIterator>
void followedBy(InputIterator groupFirst, InputIterator groupLast)
{
followedByGIDs_.clear();
for (; groupFirst != groupLast; ++groupFirst) {
followedByGIDs_.push_back((*groupFirst)->getGID());
}
}
const std::vector<a2_gid_t>& followedBy() const { return followedByGIDs_; }
void following(a2_gid_t gid) { followingGID_ = gid; }
a2_gid_t following() const { return followingGID_; }
void belongsTo(a2_gid_t gid) { belongsToGID_ = gid; }
a2_gid_t belongsTo() const { return belongsToGID_; }
void setRequestGroupMan(RequestGroupMan* requestGroupMan)
{
requestGroupMan_ = requestGroupMan;
}
RequestGroupMan* getRequestGroupMan() { return requestGroupMan_; }
int getResumeFailureCount() const { return resumeFailureCount_; }
void increaseResumeFailureCount() { ++resumeFailureCount_; }
bool p2pInvolved() const;
void setMetadataInfo(const std::shared_ptr<MetadataInfo>& info)
{
metadataInfo_ = info;
}
const std::shared_ptr<MetadataInfo>& getMetadataInfo() const
{
return metadataInfo_;
}
int getState() const { return state_; }
void setState(int state) { state_ = state; }
bool isSeedOnlyEnabled() { return seedOnly_; }
void enableSeedOnly();
// Returns true if this download is now seeding.
bool isSeeder() const;
void setPendingOption(std::shared_ptr<Option> option);
const std::shared_ptr<Option>& getPendingOption() const
{
return pendingOption_;
}
};
} // namespace aria2
#endif // D_REQUEST_GROUP_H