mirror of
https://github.com/aria2/aria2.git
synced 2026-04-11 23:39:05 +00:00
This is desirable so that we don't have to wake up disk if control file is not changed, and it is not have to be written again. We use the same method (SHA1 hash) to check the content is the same. The limitation is the hash is stored in memory, so we have to write the first time in each session. See GH-382
344 lines
9.9 KiB
C++
344 lines
9.9 KiB
C++
/* <!-- copyright */
|
|
/*
|
|
* aria2 - The high speed download utility
|
|
*
|
|
* Copyright (C) 2010 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 --> */
|
|
#include "SessionSerializer.h"
|
|
|
|
#include <cstdio>
|
|
#include <cassert>
|
|
#include <iterator>
|
|
#include <set>
|
|
|
|
#include "RequestGroupMan.h"
|
|
#include "a2functional.h"
|
|
#include "File.h"
|
|
#include "A2STR.h"
|
|
#include "download_helper.h"
|
|
#include "Option.h"
|
|
#include "DownloadResult.h"
|
|
#include "FileEntry.h"
|
|
#include "prefs.h"
|
|
#include "util.h"
|
|
#include "array_fun.h"
|
|
#include "BufferedFile.h"
|
|
#include "OptionParser.h"
|
|
#include "OptionHandler.h"
|
|
#include "SHA1IOFile.h"
|
|
|
|
#if HAVE_ZLIB
|
|
#include "GZipFile.h"
|
|
#endif
|
|
|
|
namespace aria2 {
|
|
|
|
SessionSerializer::SessionSerializer
|
|
(RequestGroupMan* requestGroupMan)
|
|
: rgman_{requestGroupMan},
|
|
saveError_{true},
|
|
saveInProgress_{true},
|
|
saveWaiting_{true}
|
|
{}
|
|
|
|
bool SessionSerializer::save(const std::string& filename) const
|
|
{
|
|
std::string tempFilename = filename;
|
|
tempFilename += "__temp";
|
|
{
|
|
std::unique_ptr<IOFile> fp;
|
|
#if HAVE_ZLIB
|
|
if (util::endsWith(filename, ".gz")) {
|
|
fp = make_unique<GZipFile>(tempFilename.c_str(), IOFile::WRITE);
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
fp = make_unique<BufferedFile>(tempFilename.c_str(), IOFile::WRITE);
|
|
}
|
|
if(!*fp) {
|
|
return false;
|
|
}
|
|
if(!save(*fp) || fp->close() == EOF) {
|
|
return false;
|
|
}
|
|
}
|
|
return File(tempFilename).renameTo(filename);
|
|
}
|
|
|
|
namespace {
|
|
// Write 1 line of option name/value pair. This function returns true
|
|
// if it succeeds, or false.
|
|
bool writeOptionLine(IOFile& fp, PrefPtr pref,
|
|
const std::string& val)
|
|
{
|
|
size_t prefLen = strlen(pref->k);
|
|
return fp.write(" ", 1) == 1 &&
|
|
fp.write(pref->k, prefLen) == prefLen &&
|
|
fp.write("=", 1) == 1 &&
|
|
fp.write(val.c_str(), val.size()) == val.size() &&
|
|
fp.write("\n", 1) == 1;
|
|
}
|
|
} // namespace
|
|
|
|
namespace {
|
|
bool writeOption(IOFile& fp, const std::shared_ptr<Option>& op)
|
|
{
|
|
const std::shared_ptr<OptionParser>& oparser = OptionParser::getInstance();
|
|
for(size_t i = 1, len = option::countOption(); i < len; ++i) {
|
|
PrefPtr pref = option::i2p(i);
|
|
const OptionHandler* h = oparser->find(pref);
|
|
if(h && h->getInitialOption() && op->definedLocal(pref)) {
|
|
if(h->getCumulative()) {
|
|
const std::string& val = op->get(pref);
|
|
std::vector<std::string> v;
|
|
util::split(val.begin(), val.end(), std::back_inserter(v), '\n',
|
|
false, false);
|
|
for(std::vector<std::string>::const_iterator j = v.begin(),
|
|
eoj = v.end(); j != eoj; ++j) {
|
|
if(!writeOptionLine(fp, pref, *j)) {
|
|
return false;
|
|
}
|
|
}
|
|
} else {
|
|
if(!writeOptionLine(fp, pref, op->get(pref))) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
} // namespace
|
|
|
|
namespace {
|
|
template<typename T>
|
|
class Unique {
|
|
typedef T type;
|
|
struct PointerCmp {
|
|
inline bool operator()(const type* x, const type* y) {
|
|
return *x < *y;
|
|
}
|
|
};
|
|
std::set<const type*, PointerCmp> known;
|
|
public:
|
|
inline bool operator()(const type& v) {
|
|
return known.insert(&v).second;
|
|
}
|
|
};
|
|
|
|
bool writeUri(IOFile& fp, const std::string& uri)
|
|
{
|
|
return fp.write(uri.c_str(), uri.size()) == uri.size() &&
|
|
fp.write("\t", 1) == 1;
|
|
}
|
|
|
|
template<typename InputIterator, class UnaryPredicate>
|
|
bool writeUri(IOFile& fp, InputIterator first, InputIterator last,
|
|
UnaryPredicate& filter)
|
|
{
|
|
for(; first != last; ++first) {
|
|
if (!filter(*first)) {
|
|
continue;
|
|
}
|
|
if(!writeUri(fp, *first)) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
} // namespace
|
|
|
|
// The downloads whose followedBy() is empty is persisted with its
|
|
// GID without no problem. For other cases, there are several patterns.
|
|
//
|
|
// 1. magnet URI
|
|
// GID of metadata download is persisted.
|
|
// 2. URI to torrent file
|
|
// GID of torrent file download is persisted.
|
|
// 3. URI to metalink file
|
|
// GID of metalink file download is persisted.
|
|
// 4. local torrent file
|
|
// GID of torrent download itself is persisted.
|
|
// 5. local metalink file
|
|
// No GID is persisted. GID is saved but it is just a random GID.
|
|
|
|
namespace {
|
|
bool writeDownloadResult
|
|
(IOFile& fp, std::set<a2_gid_t>& metainfoCache,
|
|
const std::shared_ptr<DownloadResult>& dr)
|
|
{
|
|
const std::shared_ptr<MetadataInfo>& mi = dr->metadataInfo;
|
|
if(dr->belongsTo != 0 || (mi && mi->dataOnly())) {
|
|
return true;
|
|
}
|
|
if(!mi) {
|
|
// With --force-save option, same gid may be saved twice. (e.g.,
|
|
// Downloading .meta4 followed by its content download. First
|
|
// .meta4 download is saved and second content download is also
|
|
// saved with the same gid.)
|
|
if(metainfoCache.count(dr->gid->getNumericId()) != 0) {
|
|
return true;
|
|
} else {
|
|
metainfoCache.insert(dr->gid->getNumericId());
|
|
}
|
|
// only save first file entry
|
|
if(dr->fileEntries.empty()) {
|
|
return true;
|
|
}
|
|
const std::shared_ptr<FileEntry>& file = dr->fileEntries[0];
|
|
// Don't save download if there are no URIs.
|
|
const bool hasRemaining = !file->getRemainingUris().empty();
|
|
const bool hasSpent = !file->getSpentUris().empty();
|
|
if (!hasRemaining && !hasSpent) {
|
|
return true;
|
|
}
|
|
|
|
// Save spent URIs + remaining URIs. Remove URI in spent URI which
|
|
// also exists in remaining URIs.
|
|
{
|
|
Unique<std::string> unique;
|
|
if (hasRemaining && !writeUri(fp, file->getRemainingUris().begin(),
|
|
file->getRemainingUris().end(),
|
|
unique)) {
|
|
return false;
|
|
}
|
|
if (hasSpent && !writeUri(fp, file->getSpentUris().begin(),
|
|
file->getSpentUris().end(),
|
|
unique)) {
|
|
return false;
|
|
}
|
|
}
|
|
if(fp.write("\n", 1) != 1) {
|
|
return false;
|
|
}
|
|
if(!writeOptionLine(fp, PREF_GID, dr->gid->toHex())) {
|
|
return false;
|
|
}
|
|
} else {
|
|
if(metainfoCache.count(mi->getGID()) != 0) {
|
|
return true;
|
|
} else {
|
|
metainfoCache.insert(mi->getGID());
|
|
if (fp.write(mi->getUri().c_str(),
|
|
mi->getUri().size()) != mi->getUri().size() ||
|
|
fp.write("\n", 1) != 1) {
|
|
return false;
|
|
}
|
|
// For downloads generated by metadata (e.g., BitTorrent,
|
|
// Metalink), save gid of Metadata download.
|
|
if(!writeOptionLine(fp, PREF_GID, GroupId::toHex(mi->getGID()))) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
return writeOption(fp, dr->option);
|
|
}
|
|
} // namespace
|
|
|
|
bool SessionSerializer::save(IOFile& fp) const
|
|
{
|
|
std::set<a2_gid_t> metainfoCache;
|
|
const DownloadResultList& results = rgman_->getDownloadResults();
|
|
for(const auto& dr : results) {
|
|
if(dr->result == error_code::FINISHED ||
|
|
dr->result == error_code::REMOVED) {
|
|
if(dr->option->getAsBool(PREF_FORCE_SAVE)) {
|
|
if(!writeDownloadResult(fp, metainfoCache, dr)) {
|
|
return false;
|
|
}
|
|
} else {
|
|
continue;
|
|
}
|
|
} else if(dr->result == error_code::IN_PROGRESS) {
|
|
if(saveInProgress_) {
|
|
if(!writeDownloadResult(fp, metainfoCache, dr)) {
|
|
return false;
|
|
}
|
|
}
|
|
} else {
|
|
// error download
|
|
if(saveError_) {
|
|
if(!writeDownloadResult(fp, metainfoCache, dr)) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
{
|
|
// Save active downloads.
|
|
const RequestGroupList& groups = rgman_->getRequestGroups();
|
|
for(const auto& rg : groups) {
|
|
std::shared_ptr<DownloadResult> dr = rg->createDownloadResult();
|
|
bool stopped = dr->result == error_code::FINISHED ||
|
|
dr->result == error_code::REMOVED;
|
|
if((!stopped && saveInProgress_) ||
|
|
(stopped && dr->option->getAsBool(PREF_FORCE_SAVE))) {
|
|
if(!writeDownloadResult(fp, metainfoCache, dr)) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if(saveWaiting_) {
|
|
const RequestGroupList& groups = rgman_->getReservedGroups();
|
|
for(const auto& rg : groups) {
|
|
std::shared_ptr<DownloadResult> result = rg->createDownloadResult();
|
|
if(!writeDownloadResult(fp, metainfoCache, result)) {
|
|
return false;
|
|
}
|
|
// PREF_PAUSE was removed from option, so save it here looking
|
|
// property separately.
|
|
if(rg->isPauseRequested()) {
|
|
if(!writeOptionLine(fp, PREF_PAUSE, A2_V_TRUE)) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
std::string SessionSerializer::calculateHash() const
|
|
{
|
|
SHA1IOFile sha1io;
|
|
|
|
auto rv = save(sha1io);
|
|
|
|
if(!rv) {
|
|
return "";
|
|
}
|
|
|
|
return sha1io.digest();
|
|
}
|
|
|
|
} // namespace aria2
|