mirror of
https://github.com/RetroShare/RetroShare.git
synced 2025-01-06 05:08:09 -05:00
d43a131c04
- servers now compute the map in a separate thread - CRC32Maps are kept in cache for 30 mins - CRC32Maps requests cannot be used to overflood a server anymore since their number is limited. - Transfer modules now send keep alive packets to maintain tunnels when asking for a CRC32Map git-svn-id: http://svn.code.sf.net/p/retroshare/code/trunk@4661 b45a01b8-16f6-495d-af2f-9b41ad6348cc
322 lines
8.1 KiB
C++
322 lines
8.1 KiB
C++
#include "ftfileprovider.h"
|
|
#include "ftchunkmap.h"
|
|
|
|
#include "util/rsdir.h"
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
|
|
#ifdef WINDOWS_SYS
|
|
#include "util/rswin.h"
|
|
#endif // WINDOWS_SYS
|
|
|
|
/********
|
|
* #define DEBUG_FT_FILE_PROVIDER 1
|
|
* #define DEBUG_TRANSFERS 1 // TO GET TIMESTAMPS of DATA READING
|
|
********/
|
|
|
|
#ifdef DEBUG_TRANSFERS
|
|
#include "util/rsprint.h"
|
|
#include <iomanip>
|
|
#endif
|
|
|
|
static const time_t UPLOAD_CHUNK_MAPS_TIME = 20 ; // time to ask for a new chunkmap from uploaders in seconds.
|
|
|
|
ftFileProvider::ftFileProvider(const std::string& path, uint64_t size, const std::string& hash)
|
|
: mSize(size), hash(hash), file_name(path), fd(NULL), ftcMutex("ftFileProvider")
|
|
{
|
|
RsStackMutex stack(ftcMutex); /********** STACK LOCKED MTX ******/
|
|
|
|
#ifdef DEBUG_FT_FILE_PROVIDER
|
|
std::cout << "Creating file provider for " << hash << std::endl ;
|
|
#endif
|
|
}
|
|
|
|
ftFileProvider::~ftFileProvider()
|
|
{
|
|
#ifdef DEBUG_FT_FILE_PROVIDER
|
|
std::cout << "ftFileProvider::~ftFileProvider(): Destroying file provider for " << hash << std::endl ;
|
|
#endif
|
|
if (fd!=NULL) {
|
|
fclose(fd);
|
|
fd = NULL ;
|
|
#ifdef DEBUG_FT_FILE_PROVIDER
|
|
std::cout << "ftFileProvider::~ftFileProvider(): closed file: " << hash << std::endl ;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
bool ftFileProvider::fileOk()
|
|
{
|
|
RsStackMutex stack(ftcMutex); /********** STACK LOCKED MTX ******/
|
|
return (fd != NULL);
|
|
}
|
|
|
|
std::string ftFileProvider::getHash()
|
|
{
|
|
RsStackMutex stack(ftcMutex); /********** STACK LOCKED MTX ******/
|
|
return hash;
|
|
}
|
|
|
|
uint64_t ftFileProvider::getFileSize()
|
|
{
|
|
RsStackMutex stack(ftcMutex); /********** STACK LOCKED MTX ******/
|
|
return mSize;
|
|
}
|
|
|
|
bool ftFileProvider::FileDetails(FileInfo &info)
|
|
{
|
|
RsStackMutex stack(ftcMutex); /********** STACK LOCKED MTX ******/
|
|
info.hash = hash;
|
|
info.size = mSize;
|
|
info.path = file_name;
|
|
info.fname = RsDirUtil::getTopDir(file_name);
|
|
|
|
info.transfered = 0 ; // unused
|
|
info.lastTS = 0;
|
|
info.status = FT_STATE_DOWNLOADING ;
|
|
|
|
info.peers.clear() ;
|
|
float total_transfer_rate = 0.0f ;
|
|
|
|
for(std::map<std::string,PeerUploadInfo>::const_iterator it(uploading_peers.begin());it!=uploading_peers.end();++it)
|
|
{
|
|
TransferInfo inf ;
|
|
inf.peerId = it->first ;
|
|
inf.status = FT_STATE_DOWNLOADING ;
|
|
inf.name = info.fname ;
|
|
inf.transfered = it->second.req_loc ;
|
|
|
|
inf.tfRate = it->second.transfer_rate/1024.0 ;
|
|
total_transfer_rate += it->second.transfer_rate ;
|
|
info.lastTS = std::max(info.lastTS,it->second.lastTS);
|
|
|
|
info.peers.push_back(inf) ;
|
|
}
|
|
info.tfRate = total_transfer_rate/1024.0 ;
|
|
|
|
/* Use req_loc / req_size to estimate data rate */
|
|
|
|
return true;
|
|
}
|
|
|
|
bool ftFileProvider::purgeOldPeers(time_t now,uint32_t max_duration)
|
|
{
|
|
RsStackMutex stack(ftcMutex); /********** STACK LOCKED MTX ******/
|
|
|
|
#ifdef DEBUG_FT_FILE_PROVIDER
|
|
std::cerr << "ftFileProvider::purgeOldPeers(): " << (void*)this << ": examining peers." << std::endl ;
|
|
#endif
|
|
bool ret = true ;
|
|
for(std::map<std::string,PeerUploadInfo>::iterator it(uploading_peers.begin());it!=uploading_peers.end();)
|
|
if( (*it).second.lastTS+max_duration < (uint32_t)now)
|
|
{
|
|
#ifdef DEBUG_FT_FILE_PROVIDER
|
|
std::cerr << "ftFileProvider::purgeOldPeers(): " << (void*)this << ": peer " << it->first << " is too old. Removing." << std::endl ;
|
|
#endif
|
|
std::map<std::string,PeerUploadInfo>::iterator tmp = it ;
|
|
++tmp ;
|
|
uploading_peers.erase(it) ;
|
|
it=tmp ;
|
|
}
|
|
else
|
|
{
|
|
#ifdef DEBUG_FT_FILE_PROVIDER
|
|
std::cerr << "ftFileProvider::purgeOldPeers(): " << (void*)this << ": peer " << it->first << " will be kept." << std::endl ;
|
|
#endif
|
|
ret = false ;
|
|
++it ;
|
|
}
|
|
return ret ;
|
|
}
|
|
|
|
void ftFileProvider::getAvailabilityMap(CompressedChunkMap& cmap)
|
|
{
|
|
// We are here because the file we deal with is complete. So we return a plain map.
|
|
//
|
|
ChunkMap::buildPlainMap(mSize,cmap) ;
|
|
}
|
|
|
|
|
|
bool ftFileProvider::getFileData(const std::string& peer_id,uint64_t offset, uint32_t &chunk_size, void *data)
|
|
{
|
|
/* dodgey checking outside of mutex...
|
|
* much check again inside FileAttrs().
|
|
*/
|
|
if (fd == NULL)
|
|
if (!initializeFileAttrs())
|
|
return false;
|
|
|
|
RsStackMutex stack(ftcMutex); /********** STACK LOCKED MTX ******/
|
|
|
|
/*
|
|
* FIXME: Warning of comparison between unsigned and signed int?
|
|
*/
|
|
|
|
uint32_t data_size = chunk_size;
|
|
uint64_t base_loc = offset;
|
|
|
|
if (base_loc + data_size > mSize)
|
|
{
|
|
data_size = mSize - base_loc;
|
|
chunk_size = mSize - base_loc;
|
|
std::cerr <<"Chunk Size greater than total file size, adjusting chunk size " << data_size << std::endl;
|
|
}
|
|
|
|
if(data_size > 0 && data != NULL)
|
|
{
|
|
/*
|
|
* seek for base_loc
|
|
*/
|
|
fseeko64(fd, base_loc, SEEK_SET);
|
|
|
|
// Data space allocated by caller.
|
|
//void *data = malloc(chunk_size);
|
|
|
|
/*
|
|
* read the data
|
|
*/
|
|
|
|
if (1 != fread(data, data_size, 1, fd))
|
|
{
|
|
#ifdef DEBUG_FT_FILE_PROVIDER
|
|
std::cerr << "ftFileProvider::getFileData() Failed to get data. Data_size=" << data_size << ", base_loc=" << base_loc << " !" << std::endl;
|
|
#endif
|
|
//free(data); No!! It's already freed upwards in ftDataMultiplex::locked_handleServerRequest()
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Update status of ftFileStatus to reflect last usage (for GUI display)
|
|
* We need to store.
|
|
* (a) Id,
|
|
* (b) Offset,
|
|
* (c) Size,
|
|
* (d) timestamp
|
|
*/
|
|
|
|
// This creates the peer info, and updates it.
|
|
//
|
|
time_t now = time(NULL) ;
|
|
uploading_peers[peer_id].updateStatus(offset,data_size,now) ;
|
|
|
|
#ifdef DEBUG_TRANSFERS
|
|
std::cerr << "ftFileProvider::getFileData() ";
|
|
std::cerr << " at " << RsUtil::AccurateTimeString();
|
|
std::cerr << " hash: " << hash;
|
|
std::cerr << " for peerId: " << peer_id;
|
|
std::cerr << " offset: " << offset;
|
|
std::cerr << " chunkSize: " << chunk_size;
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
}
|
|
else
|
|
{
|
|
std::cerr << "No data to read, or NULL buffer used" << std::endl;
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
void ftFileProvider::PeerUploadInfo::updateStatus(uint64_t offset,uint32_t data_size,time_t now)
|
|
{
|
|
lastTS = now ;
|
|
long int diff = (long int)now - (long int)lastTS_t ; // in bytes/s. Average over multiple samples
|
|
|
|
#ifdef DEBUG_FT_FILE_PROVIDER
|
|
std::cout << "diff = " << diff << std::endl ;
|
|
#endif
|
|
|
|
if(diff > 3)
|
|
{
|
|
transfer_rate = total_size / (float)diff ;
|
|
#ifdef DEBUG_FT_FILE_PROVIDER
|
|
std::cout << "updated TR = " << transfer_rate << ", total_size=" << total_size << std::endl ;
|
|
#endif
|
|
lastTS_t = now ;
|
|
total_size = 0 ;
|
|
}
|
|
|
|
req_loc = offset;
|
|
req_size = data_size;
|
|
total_size += req_size ;
|
|
}
|
|
|
|
void ftFileProvider::setClientMap(const std::string& peer_id,const CompressedChunkMap& cmap)
|
|
{
|
|
RsStackMutex stack(ftcMutex); /********** STACK LOCKED MTX ******/
|
|
|
|
// Create by default.
|
|
uploading_peers[peer_id].client_chunk_map = cmap ;
|
|
uploading_peers[peer_id].client_chunk_map_stamp = time(NULL) ;
|
|
}
|
|
|
|
void ftFileProvider::getClientMap(const std::string& peer_id,CompressedChunkMap& cmap,bool& map_is_too_old)
|
|
{
|
|
RsStackMutex stack(ftcMutex); /********** STACK LOCKED MTX ******/
|
|
|
|
PeerUploadInfo& pui(uploading_peers[peer_id]) ;
|
|
|
|
time_t now = time(NULL) ;
|
|
|
|
if(now - pui.client_chunk_map_stamp > UPLOAD_CHUNK_MAPS_TIME)
|
|
{
|
|
map_is_too_old = true ;
|
|
pui.client_chunk_map_stamp = now ; // to avoid re-asking before the TTL
|
|
}
|
|
else
|
|
map_is_too_old = false ;
|
|
|
|
cmap = pui.client_chunk_map;
|
|
}
|
|
|
|
int ftFileProvider::initializeFileAttrs()
|
|
{
|
|
#ifdef DEBUG_FT_FILE_PROVIDER
|
|
std::cerr << "ftFileProvider::initializeFileAttrs() Filename: " << file_name << std::endl;
|
|
#endif
|
|
|
|
RsStackMutex stack(ftcMutex); /********** STACK LOCKED MTX ******/
|
|
if (fd)
|
|
return 1;
|
|
|
|
/*
|
|
* check if the file exists
|
|
*/
|
|
|
|
{
|
|
#ifdef DEBUG_FT_FILE_PROVIDER
|
|
std::cerr << "ftFileProvider::initializeFileAttrs() trying (r+b) " << std::endl;
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
* attempt to open file
|
|
*/
|
|
|
|
fd = RsDirUtil::rs_fopen(file_name.c_str(), "r+b");
|
|
if (!fd)
|
|
{
|
|
std::cerr << "ftFileProvider::initializeFileAttrs() Failed to open (r+b): ";
|
|
std::cerr << file_name << std::endl;
|
|
|
|
/* try opening read only */
|
|
fd = RsDirUtil::rs_fopen(file_name.c_str(), "rb");
|
|
if (!fd)
|
|
{
|
|
std::cerr << "ftFileProvider::initializeFileAttrs() Failed to open (rb): ";
|
|
std::cerr << file_name << std::endl;
|
|
|
|
/* try opening read only */
|
|
return 0;
|
|
}
|
|
}
|
|
#ifdef DEBUG_FT_FILE_PROVIDER
|
|
std::cerr << "ftFileProvider:: openned file " << file_name << std::endl ;
|
|
#endif
|
|
|
|
return 1;
|
|
}
|
|
|
|
|