/* */ #include "DHTAutoSaveCommand.h" #include #include #include "DHTRoutingTable.h" #include "DHTNode.h" #include "File.h" #include "DHTRoutingTableSerializer.h" #include "RecoverableException.h" #include "DownloadEngine.h" #include "DHTBucket.h" #include "RequestGroupMan.h" #include "prefs.h" #include "Option.h" #include "message.h" #include "Logger.h" #include "LogFactory.h" #include "a2functional.h" #include "FileEntry.h" #include "DlAbortEx.h" #include "fmt.h" namespace aria2 { DHTAutoSaveCommand::DHTAutoSaveCommand (cuid_t cuid, DownloadEngine* e, int family, time_t interval) : TimeBasedCommand(cuid, e, interval), family_(family) {} DHTAutoSaveCommand::~DHTAutoSaveCommand() {} void DHTAutoSaveCommand::preProcess() { if(getDownloadEngine()->getRequestGroupMan()->downloadFinished() || getDownloadEngine()->isHaltRequested()) { save(); enableExit(); } } void DHTAutoSaveCommand::process() { save(); } void DHTAutoSaveCommand::save() { std::string dhtFile = getDownloadEngine()->getOption()-> get(family_ == AF_INET? PREF_DHT_FILE_PATH : PREF_DHT_FILE_PATH6); A2_LOG_INFO(fmt("Saving DHT routing table to %s.", dhtFile.c_str())); File tempFile(dhtFile+"__temp"); // Removing tempFile is unnecessary because the file is truncated on // open. But the bug in 1.10.4 creates directory for this path. // Because it is directory, opening directory as file fails. So we // first remove it here. tempFile.remove(); File(tempFile.getDirname()).mkdirs(); std::vector > nodes; std::vector > buckets; routingTable_->getBuckets(buckets); for(std::vector >::const_iterator i = buckets.begin(), eoi = buckets.end(); i != eoi; ++i) { const SharedHandle& bucket = *i; std::vector > goodNodes; bucket->getGoodNodes(goodNodes); nodes.insert(nodes.end(), goodNodes.begin(), goodNodes.end()); } DHTRoutingTableSerializer serializer(family_); serializer.setLocalNode(localNode_); serializer.setNodes(nodes); try { { std::ofstream o(tempFile.getPath().c_str(), std::ios::out|std::ios::binary); if(!o) { throw DL_ABORT_EX (fmt("Failed to save DHT routing table to %s.", dhtFile.c_str())); } serializer.serialize(o); } if(!tempFile.renameTo(dhtFile)) { A2_LOG_ERROR(fmt("Cannot move file from %s to %s.", tempFile.getPath().c_str(), dhtFile.c_str())); } } catch(RecoverableException& e) { A2_LOG_ERROR_EX(fmt("Exception caught while saving DHT routing table to %s", dhtFile.c_str()), e); } } void DHTAutoSaveCommand::setLocalNode(const SharedHandle& localNode) { localNode_ = localNode; } void DHTAutoSaveCommand::setRoutingTable (const SharedHandle& routingTable) { routingTable_ = routingTable; } } // namespace aria2