aria2/src/SessionSerializer.cc
2019-09-15 11:26:48 +09:00

368 lines
11 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) const
{
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,
bool pauseRequested)
{
const std::shared_ptr<MetadataInfo>& mi = dr->metadataInfo;
if (dr->belongsTo != 0 || (mi && mi->dataOnly()) || !dr->followedBy.empty()) {
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;
}
}
}
// PREF_PAUSE was removed from option, so save it here looking
// property separately.
if (pauseRequested) {
if (!writeOptionLine(fp, PREF_PAUSE, A2_V_TRUE)) {
return false;
}
}
return writeOption(fp, dr->option);
}
} // namespace
namespace {
template <typename InputIt>
bool saveDownloadResult(IOFile& fp, std::set<a2_gid_t>& metainfoCache,
InputIt first, InputIt last, bool saveInProgress,
bool saveError)
{
for (; first != last; ++first) {
const auto& dr = *first;
auto save = false;
switch (dr->result) {
case error_code::FINISHED:
case error_code::REMOVED:
save = dr->option->getAsBool(PREF_FORCE_SAVE);
break;
case error_code::IN_PROGRESS:
save = saveInProgress;
break;
case error_code::RESOURCE_NOT_FOUND:
case error_code::MAX_FILE_NOT_FOUND:
save = saveError && dr->option->getAsBool(PREF_SAVE_NOT_FOUND);
break;
default:
save = saveError;
break;
}
if (save && !writeDownloadResult(fp, metainfoCache, dr, false)) {
return false;
}
}
return true;
}
} // namespace
bool SessionSerializer::save(IOFile& fp) const
{
std::set<a2_gid_t> metainfoCache;
const auto& unfinishedResults = rgman_->getUnfinishedDownloadResult();
if (!saveDownloadResult(fp, metainfoCache, std::begin(unfinishedResults),
std::end(unfinishedResults), saveInProgress_,
saveError_)) {
return false;
}
const auto& results = rgman_->getDownloadResults();
if (!saveDownloadResult(fp, metainfoCache, std::begin(results),
std::end(results), saveInProgress_, saveError_)) {
return false;
}
{
// Save active downloads.
const RequestGroupList& groups = rgman_->getRequestGroups();
for (const auto& rg : groups) {
auto 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,
rg->isPauseRequested())) {
return false;
}
}
}
}
if (saveWaiting_) {
const auto& groups = rgman_->getReservedGroups();
for (const auto& rg : groups) {
auto result = rg->createDownloadResult();
if (!writeDownloadResult(fp, metainfoCache, result,
rg->isPauseRequested())) {
return false;
}
}
}
return true;
}
std::string SessionSerializer::calculateHash() const
{
SHA1IOFile sha1io;
auto rv = save(sha1io);
if (!rv) {
return "";
}
return sha1io.digest();
}
} // namespace aria2