/******************************************************************************* * libretroshare/src/ft: ftextralist.cc * * * * libretroshare: retroshare core library * * * * Copyright (C) 2008 Robert Fernie * * Copyright (C) 2018-2020 Gioacchino Mazzurco * * * * This program is free software: you can redistribute it and/or modify * * it under the terms of the GNU Lesser General Public License as * * published by the Free Software Foundation, either version 3 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 Lesser General Public License for more details. * * * * You should have received a copy of the GNU Lesser General Public License * * along with this program. If not, see . * * * *******************************************************************************/ #include #include #ifdef WINDOWS_SYS #include "util/rswin.h" #endif #include #include #include "ft/ftextralist.h" #include "rsitems/rsconfigitems.h" #include "util/rsdir.h" #include "util/rstime.h" #include #include /* for (u)sleep() */ #include "util/rstime.h" /****** * #define DEBUG_ELIST 1 *****/ ftExtraList::ftExtraList() :p3Config(), extMutex("p3Config") { cleanup = 0; return; } void ftExtraList::threadTick() { bool todo = false; rstime_t now = time(NULL); { RsStackMutex stack(extMutex); todo = (mToHash.size() > 0); } if (todo) { /* Hash a file */ hashAFile(); /* microsleep */ rstime::rs_usleep(10); } else { /* cleanup */ if (cleanup < now) { cleanupOldFiles(); cleanup = now + CLEANUP_PERIOD; } /* sleep */ #ifdef WIN32 Sleep(1000); #else sleep(1); #endif } } void ftExtraList::hashAFile() { #ifdef DEBUG_ELIST std::cerr << "ftExtraList::hashAFile()"; std::cerr << std::endl; #endif /* extract entry from the queue */ FileDetails details; { RS_STACK_MUTEX(extMutex); if (mToHash.empty()) return; details = mToHash.front(); mToHash.pop_front(); } #ifdef DEBUG_ELIST std::cerr << "Hashing: " << details.info.path; std::cerr << std::endl; #endif /* hash it! */ std::string name, hash; //uint64_t size; if (RsDirUtil::hashFile(details.info.path, details.info.fname, details.info.hash, details.info.size)) { RS_STACK_MUTEX(extMutex); /* stick it in the available queue */ mFiles[details.info.hash] = details; mHashOfHash[makeEncryptedHash(details.info.hash)] = details.info.hash ; /* add to the path->hash map */ mHashedList[details.info.path] = details.info.hash; IndicateConfigChanged(); auto ev = std::make_shared(); ev->mEventCode = RsSharedDirectoriesEventCode::EXTRA_LIST_FILE_ADDED; if(rsEvents) rsEvents->postEvent(ev); } } /*** * If the File is alreay Hashed, then just add it in. **/ bool ftExtraList::addExtraFile(std::string path, const RsFileHash& hash, uint64_t size, uint32_t period, TransferRequestFlags flags) { #ifdef DEBUG_ELIST std::cerr << "ftExtraList::addExtraFile() path: " << path; std::cerr << " hash: " << hash; std::cerr << " size: " << size; std::cerr << " period: " << period; std::cerr << " flags: " << flags; std::cerr << std::endl; #endif RS_STACK_MUTEX(extMutex); FileDetails details; details.info.path = path; details.info.fname = RsDirUtil::getTopDir(path); details.info.hash = hash; details.info.size = size; details.info.age = time(NULL) + period; /* if time > this... cleanup */ details.info.transfer_info_flags = flags ; /* stick it in the available queue */ mFiles[details.info.hash] = details; mHashOfHash[makeEncryptedHash(details.info.hash)] = details.info.hash ; IndicateConfigChanged(); return true; } bool ftExtraList::removeExtraFile(const RsFileHash& hash) { /* remove unused parameter warnings */ #ifdef DEBUG_ELIST std::cerr << "ftExtraList::removeExtraFile()"; std::cerr << " hash: " << hash; std::cerr << " flags: " << flags; std::cerr << std::endl; #endif RS_STACK_MUTEX(extMutex); mHashOfHash.erase(makeEncryptedHash(hash)) ; std::map::iterator it; it = mFiles.find(hash); if (it == mFiles.end()) { return false; } mFiles.erase(it); IndicateConfigChanged(); if(rsEvents) { auto ev = std::make_shared(); ev->mEventCode = RsSharedDirectoriesEventCode::EXTRA_LIST_FILE_REMOVED; rsEvents->postEvent(ev); } return true; } bool ftExtraList::moveExtraFile(std::string fname, const RsFileHash &hash, uint64_t /*size*/, std::string destpath) { RsStackMutex stack(extMutex); std::map::iterator it; it = mFiles.find(hash); if (it == mFiles.end()) { return false; } std::string path = destpath + '/' + fname; if (RsDirUtil::renameFile(it->second.info.path, path)) { /* rename */ it->second.info.path = path; it->second.info.fname = fname; IndicateConfigChanged(); } return true; } bool ftExtraList::cleanupOldFiles() { #ifdef DEBUG_ELIST std::cerr << "ftExtraList::cleanupOldFiles()"; std::cerr << std::endl; #endif RS_STACK_MUTEX(extMutex); rstime_t now = time(NULL); std::list toRemove; for( std::map::iterator it = mFiles.begin(); it != mFiles.end(); ++it) /* check timestamps */ if ((rstime_t)it->second.info.age < now) toRemove.push_back(it->first); if (toRemove.size() > 0) { std::map::iterator it; /* remove items */ for(std::list::iterator rit = toRemove.begin(); rit != toRemove.end(); ++rit) { if (mFiles.end() != (it = mFiles.find(*rit))) mFiles.erase(it); mHashOfHash.erase(makeEncryptedHash(*rit)); } IndicateConfigChanged(); } return true; } bool ftExtraList::hashExtraFile( std::string path, uint32_t period, TransferRequestFlags flags ) { constexpr rstime_t max_int = std::numeric_limits::max(); const rstime_t now = time(nullptr); const rstime_t timeOut = now + period; if(timeOut > max_int) { /* Under the hood period is stored as int FileInfo::age so we do this * check here to detect 2038 year problem * https://en.wikipedia.org/wiki/Year_2038_problem */ RsErr() << __PRETTY_FUNCTION__ << " period: " << period << " > " << max_int - now << std::errc::value_too_large << std::endl; return false; } if(!RsDirUtil::fileExists(path)) { RsErr() << __PRETTY_FUNCTION__ << " path: " << path << std::errc::no_such_file_or_directory << std::endl; return false; } if(RsDirUtil::checkDirectory(path)) { RsErr() << __PRETTY_FUNCTION__ << " path: " << path << std::errc::is_a_directory << std::endl; return false; } FileDetails details(path, period, flags); details.info.age = static_cast(timeOut); { RS_STACK_MUTEX(extMutex); mToHash.push_back(details); /* add into queue */ } return true; } bool ftExtraList::hashExtraFileDone(std::string path, FileInfo &info) { #ifdef DEBUG_ELIST std::cerr << "ftExtraList::hashExtraFileDone()"; std::cerr << std::endl; #endif RsFileHash hash; { /* Find in the path->hash map */ RS_STACK_MUTEX(extMutex); std::map::iterator it; if (mHashedList.end() == (it = mHashedList.find(path))) { return false; } hash = it->second; } return search(hash, FileSearchFlags(0), info); } /*** * Search Function - used by File Transfer * **/ bool ftExtraList::search(const RsFileHash &hash, FileSearchFlags /*hintflags*/, FileInfo &info) const { #ifdef DEBUG_ELIST std::cerr << "ftExtraList::search() hash=" << hash ; #endif /* find hash */ std::map::const_iterator fit; if (mFiles.end() == (fit = mFiles.find(hash))) { #ifdef DEBUG_ELIST std::cerr << " not found in mFiles. Trying encrypted... " ; #endif // File not found. We try to look for encrypted hash. std::map::const_iterator hit = mHashOfHash.find(hash) ; if(hit == mHashOfHash.end()) { #ifdef DEBUG_ELIST std::cerr << " not found." << std::endl; #endif return false; } #ifdef DEBUG_ELIST std::cerr << " found! Reaching data..." ; #endif fit = mFiles.find(hit->second) ; if(fit == mFiles.end()) // not found. This is an error. { #ifdef DEBUG_ELIST std::cerr << " no data. Returning false." << std::endl; #endif return false ; } #ifdef DEBUG_ELIST std::cerr << " ok! Accepting encrypted transfer." << std::endl; #endif info = fit->second.info; info.storage_permission_flags = FileStorageFlags(DIR_FLAGS_ANONYMOUS_DOWNLOAD) ; info.transfer_info_flags |= RS_FILE_REQ_ENCRYPTED ; } else { #ifdef DEBUG_ELIST std::cerr << " found! Accepting direct transfer" << std::endl; #endif info = fit->second.info; // Unencrypted file transfer: We only allow direct transfers. This is not exactly secure since another friend can // swarm the file. But the hash being kept secret, there's no risk here. // info.storage_permission_flags = FileStorageFlags(DIR_FLAGS_BROWSABLE) ; } if(info.transfer_info_flags & RS_FILE_REQ_ANONYMOUS_ROUTING) info.storage_permission_flags |= DIR_FLAGS_ANONYMOUS_DOWNLOAD ; return true; } RsFileHash ftExtraList::makeEncryptedHash(const RsFileHash& hash) { return RsDirUtil::sha1sum(hash.toByteArray(),hash.SIZE_IN_BYTES); } /*** * Configuration - store extra files. * **/ RsSerialiser *ftExtraList::setupSerialiser() { RsSerialiser *rss = new RsSerialiser(); /* add in the types we need! */ rss->addSerialType(new RsFileConfigSerialiser()); return rss; } bool ftExtraList::saveList(bool &cleanup, std::list& sList) { cleanup = true; /* called after each item is added */ /* create a list of fileitems with * age used to specify its timeout. */ #ifdef DEBUG_ELIST std::cerr << "ftExtraList::saveList()"; std::cerr << std::endl; #endif RS_STACK_MUTEX(extMutex); std::map::const_iterator it; for(it = mFiles.begin(); it != mFiles.end(); ++it) { RsFileConfigItem *fi = new RsFileConfigItem(); fi->file.path = (it->second).info.path; fi->file.name = (it->second).info.fname; fi->file.hash = (it->second).info.hash; fi->file.filesize = (it->second).info.size; fi->file.age = (it->second).info.age; fi->flags = (it->second).info.transfer_info_flags.toUInt32(); sList.push_back(fi); } return true; } bool ftExtraList::loadList(std::list& load) { /* for each item, check it exists .... * - remove any that are dead (or flag?) */ #ifdef DEBUG_ELIST std::cerr << "ftExtraList::loadList()"; std::cerr << std::endl; #endif rstime_t ts = time(NULL); std::list::iterator it; for(it = load.begin(); it != load.end(); ++it) { RsFileConfigItem *fi = dynamic_cast(*it); if (!fi) { delete (*it); continue; } /* open file */ FILE *fd = RsDirUtil::rs_fopen(fi->file.path.c_str(), "rb"); if (fd == NULL) { delete (*it); continue; } fclose(fd); fd = NULL ; if (ts > (rstime_t)fi->file.age) { /* too old */ delete (*it); continue ; } /* add into system */ FileDetails file; RS_STACK_MUTEX(extMutex); FileDetails details; details.info.path = fi->file.path; details.info.fname = fi->file.name; details.info.hash = fi->file.hash; details.info.size = fi->file.filesize; details.info.age = fi->file.age; /* time that we remove it. */ details.info.transfer_info_flags = TransferRequestFlags(fi->flags); /* stick it in the available queue */ mFiles[details.info.hash] = details; mHashOfHash[makeEncryptedHash(details.info.hash)] = details.info.hash ; delete (*it); /* short sleep */ rstime::rs_usleep(1000) ; } load.clear() ; return true; } void ftExtraList::getExtraFileList(std::vector& files) const { RS_STACK_MUTEX(extMutex); files.clear(); for(auto it(mFiles.begin());it!=mFiles.end();++it) files.push_back(it->second.info); }