mirror of
https://github.com/RetroShare/RetroShare.git
synced 2024-12-31 18:36:24 -05:00
539 lines
13 KiB
C++
539 lines
13 KiB
C++
/*******************************************************************************
|
|
* libretroshare/src/ft: ftextralist.cc *
|
|
* *
|
|
* libretroshare: retroshare core library *
|
|
* *
|
|
* Copyright (C) 2008 Robert Fernie <retroshare@lunamutt.com> *
|
|
* Copyright (C) 2018-2020 Gioacchino Mazzurco <gio@eigenlab.org> *
|
|
* *
|
|
* 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 <https://www.gnu.org/licenses/>. *
|
|
* *
|
|
*******************************************************************************/
|
|
|
|
#include <limits>
|
|
#include <system_error>
|
|
|
|
#ifdef WINDOWS_SYS
|
|
#include "util/rswin.h"
|
|
#endif
|
|
|
|
#include <retroshare/rstypes.h>
|
|
#include <retroshare/rsfiles.h>
|
|
#include "ft/ftextralist.h"
|
|
#include "rsitems/rsconfigitems.h"
|
|
#include "util/rsdir.h"
|
|
#include "util/rstime.h"
|
|
#include <stdio.h>
|
|
#include <unistd.h> /* 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<RsSharedDirectoriesEvent>();
|
|
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<RsFileHash, FileDetails>::iterator it;
|
|
it = mFiles.find(hash);
|
|
if (it == mFiles.end())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
mFiles.erase(it);
|
|
|
|
IndicateConfigChanged();
|
|
|
|
if(rsEvents)
|
|
{
|
|
auto ev = std::make_shared<RsSharedDirectoriesEvent>();
|
|
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<RsFileHash, FileDetails>::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<RsFileHash> toRemove;
|
|
|
|
for( std::map<RsFileHash, FileDetails>::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<RsFileHash, FileDetails>::iterator it;
|
|
|
|
/* remove items */
|
|
for(std::list<RsFileHash>::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<int>::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<int>(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<std::string, RsFileHash>::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<RsFileHash, FileDetails>::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<RsFileHash,RsFileHash>::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<RsItem *>& 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<RsFileHash, FileDetails>::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<RsItem *>& 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<RsItem *>::iterator it;
|
|
for(it = load.begin(); it != load.end(); ++it)
|
|
{
|
|
|
|
RsFileConfigItem *fi = dynamic_cast<RsFileConfigItem *>(*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<FileInfo>& files) const
|
|
{
|
|
RS_STACK_MUTEX(extMutex);
|
|
|
|
files.clear();
|
|
|
|
for(auto it(mFiles.begin());it!=mFiles.end();++it)
|
|
files.push_back(it->second.info);
|
|
}
|