2008-07-23 18:01:59 -04:00
|
|
|
/*
|
|
|
|
* libretroshare/src/ft: ftcontroller.cc
|
|
|
|
*
|
|
|
|
* File Transfer for RetroShare.
|
|
|
|
*
|
|
|
|
* Copyright 2008 by Robert Fernie.
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Library General Public
|
|
|
|
* License Version 2 as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This library 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
|
|
|
|
* Library General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Library General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
|
|
|
|
* USA.
|
|
|
|
*
|
|
|
|
* Please report all bugs and problems to "retroshare@lunamutt.com".
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* ftController
|
|
|
|
*
|
|
|
|
* Top level download controller.
|
|
|
|
*
|
|
|
|
* inherits configuration (save downloading files)
|
|
|
|
* inherits pqiMonitor (knows which peers are online).
|
|
|
|
* inherits CacheTransfer (transfers cache files too)
|
|
|
|
* inherits RsThread (to control transfers)
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2008-08-09 13:03:24 -04:00
|
|
|
#include "ft/ftcontroller.h"
|
|
|
|
|
|
|
|
#include "ft/ftfilecreator.h"
|
|
|
|
#include "ft/fttransfermodule.h"
|
|
|
|
#include "ft/ftsearch.h"
|
|
|
|
#include "ft/ftdatamultiplex.h"
|
2008-11-09 17:17:20 -05:00
|
|
|
#include "ft/ftextralist.h"
|
2008-08-09 13:03:24 -04:00
|
|
|
|
|
|
|
#include "util/rsdir.h"
|
2008-07-23 18:01:59 -04:00
|
|
|
|
2008-08-09 13:03:24 -04:00
|
|
|
#include "pqi/p3connmgr.h"
|
2008-07-23 18:01:59 -04:00
|
|
|
|
2008-11-15 15:00:29 -05:00
|
|
|
#include "serialiser/rsconfigitems.h"
|
2008-08-09 13:03:24 -04:00
|
|
|
|
|
|
|
|
2008-11-15 15:00:29 -05:00
|
|
|
#define CONTROL_DEBUG 1
|
2008-08-09 13:03:24 -04:00
|
|
|
|
2008-08-17 11:23:11 -04:00
|
|
|
ftFileControl::ftFileControl()
|
|
|
|
:mTransfer(NULL), mCreator(NULL),
|
|
|
|
mState(0), mSize(0), mFlags(0)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2008-10-29 16:58:23 -04:00
|
|
|
ftFileControl::ftFileControl(std::string fname,
|
|
|
|
std::string tmppath, std::string dest,
|
|
|
|
uint64_t size, std::string hash, uint32_t flags,
|
|
|
|
ftFileCreator *fc, ftTransferModule *tm, uint32_t cb)
|
|
|
|
:mName(fname), mCurrentPath(tmppath), mDestination(dest),
|
|
|
|
mTransfer(tm), mCreator(fc), mState(0), mHash(hash),
|
|
|
|
mSize(size), mFlags(0), mDoCallback(false), mCallbackCode(cb)
|
2008-08-17 11:23:11 -04:00
|
|
|
{
|
2008-10-29 16:58:23 -04:00
|
|
|
if (cb)
|
|
|
|
mDoCallback = true;
|
2008-08-17 11:23:11 -04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2008-08-09 13:03:24 -04:00
|
|
|
ftController::ftController(CacheStrapper *cs, ftDataMultiplex *dm, std::string configDir)
|
2008-11-15 15:00:29 -05:00
|
|
|
:CacheTransfer(cs), p3Config(CONFIG_TYPE_FT_CONTROL), mDataplex(dm)
|
2008-07-23 18:01:59 -04:00
|
|
|
{
|
2008-08-09 13:03:24 -04:00
|
|
|
/* TODO */
|
2008-07-23 18:01:59 -04:00
|
|
|
}
|
|
|
|
|
2008-11-09 17:17:20 -05:00
|
|
|
void ftController::setFtSearchNExtra(ftSearch *search, ftExtraList *list)
|
2008-08-09 13:03:24 -04:00
|
|
|
{
|
|
|
|
mSearch = search;
|
2008-11-09 17:17:20 -05:00
|
|
|
mExtraList = list;
|
2008-08-09 13:03:24 -04:00
|
|
|
}
|
2008-07-23 18:01:59 -04:00
|
|
|
|
2008-08-09 13:03:24 -04:00
|
|
|
void ftController::run()
|
2008-07-23 18:01:59 -04:00
|
|
|
{
|
|
|
|
/* check the queues */
|
2008-08-17 11:23:11 -04:00
|
|
|
|
2008-08-21 17:30:59 -04:00
|
|
|
while(1)
|
2008-08-17 11:23:11 -04:00
|
|
|
{
|
2008-08-25 16:03:39 -04:00
|
|
|
#ifdef WIN32
|
|
|
|
Sleep(1000);
|
|
|
|
#else
|
2008-08-21 17:30:59 -04:00
|
|
|
sleep(1);
|
2008-08-25 16:03:39 -04:00
|
|
|
#endif
|
2008-08-21 17:30:59 -04:00
|
|
|
|
2008-11-02 06:38:11 -05:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
//std::cerr << "ftController::run()";
|
|
|
|
//std::cerr << std::endl;
|
|
|
|
#endif
|
2008-08-17 11:23:11 -04:00
|
|
|
|
2008-08-21 17:30:59 -04:00
|
|
|
/* tick the transferModules */
|
2008-10-29 16:58:23 -04:00
|
|
|
std::list<std::string> done;
|
|
|
|
std::list<std::string>::iterator it;
|
2008-08-21 17:30:59 -04:00
|
|
|
{
|
2008-10-29 16:58:23 -04:00
|
|
|
RsStackMutex stack(ctrlMutex); /******* LOCKED ********/
|
|
|
|
|
|
|
|
std::map<std::string, ftFileControl>::iterator it;
|
|
|
|
for(it = mDownloads.begin(); it != mDownloads.end(); it++)
|
|
|
|
{
|
2008-08-21 17:30:59 -04:00
|
|
|
std::cerr << "\tTicking: " << it->first;
|
|
|
|
std::cerr << std::endl;
|
|
|
|
|
2008-10-29 16:58:23 -04:00
|
|
|
if (it->second.mTransfer)
|
|
|
|
(it->second.mTransfer)->tick();
|
|
|
|
}
|
2008-08-21 17:30:59 -04:00
|
|
|
}
|
2008-10-29 16:58:23 -04:00
|
|
|
|
|
|
|
RsStackMutex stack2(doneMutex);
|
|
|
|
for(it = mDone.begin(); it != mDone.end(); it++)
|
|
|
|
{
|
|
|
|
completeFile(*it);
|
|
|
|
}
|
|
|
|
mDone.clear();
|
2008-08-17 11:23:11 -04:00
|
|
|
}
|
2008-10-29 16:58:23 -04:00
|
|
|
|
2008-07-23 18:01:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Called every 10 seconds or so */
|
|
|
|
void ftController::checkDownloadQueue()
|
|
|
|
{
|
|
|
|
/* */
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2008-10-29 16:58:23 -04:00
|
|
|
bool ftController::FlagFileComplete(std::string hash)
|
2008-07-23 18:01:59 -04:00
|
|
|
{
|
2008-10-29 16:58:23 -04:00
|
|
|
RsStackMutex stack2(doneMutex);
|
|
|
|
mDone.push_back(hash);
|
2008-08-09 13:03:24 -04:00
|
|
|
|
2008-10-29 16:58:23 -04:00
|
|
|
std::cerr << "ftController:FlagFileComplete(" << hash << ")";
|
|
|
|
std::cerr << std::endl;
|
2008-08-09 13:03:24 -04:00
|
|
|
|
2008-10-29 16:58:23 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ftController::completeFile(std::string hash)
|
|
|
|
{
|
2008-11-09 17:17:20 -05:00
|
|
|
/* variables... so we can drop mutex later */
|
|
|
|
std::string path;
|
|
|
|
uint64_t size = 0;
|
|
|
|
uint32_t state = 0;
|
|
|
|
uint32_t period = 0;
|
|
|
|
uint32_t flags = 0;
|
|
|
|
|
|
|
|
bool doCallback = false;
|
|
|
|
uint32_t callbackCode = 0;
|
|
|
|
|
|
|
|
|
|
|
|
{ RsStackMutex stack(ctrlMutex); /******* LOCKED ********/
|
2008-07-23 18:01:59 -04:00
|
|
|
|
2008-10-29 16:58:23 -04:00
|
|
|
std::cerr << "ftController:completeFile(" << hash << ")";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
|
2008-07-23 18:01:59 -04:00
|
|
|
std::map<std::string, ftFileControl>::iterator it;
|
|
|
|
it = mDownloads.find(hash);
|
|
|
|
if (it == mDownloads.end())
|
|
|
|
{
|
2008-10-29 16:58:23 -04:00
|
|
|
std::cerr << "ftController:completeFile(" << hash << ")";
|
|
|
|
std::cerr << " Not Found!";
|
|
|
|
std::cerr << std::endl;
|
2008-07-23 18:01:59 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check if finished */
|
2008-10-29 16:58:23 -04:00
|
|
|
if (!(it->second).mCreator->finished())
|
2008-07-23 18:01:59 -04:00
|
|
|
{
|
|
|
|
/* not done! */
|
2008-10-29 16:58:23 -04:00
|
|
|
std::cerr << "ftController:completeFile(" << hash << ")";
|
|
|
|
std::cerr << " Transfer Not Done";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
|
|
|
|
std::cerr << "FileSize: ";
|
|
|
|
std::cerr << (it->second).mCreator->getFileSize();
|
|
|
|
std::cerr << " and Recvd: ";
|
|
|
|
std::cerr << (it->second).mCreator->getRecvd();
|
|
|
|
|
|
|
|
return false;
|
2008-07-23 18:01:59 -04:00
|
|
|
}
|
2008-09-11 06:33:43 -04:00
|
|
|
|
2008-07-23 18:01:59 -04:00
|
|
|
|
|
|
|
ftFileControl *fc = &(it->second);
|
|
|
|
|
|
|
|
/* done - cleanup */
|
2008-10-29 16:58:23 -04:00
|
|
|
|
2008-08-09 13:03:24 -04:00
|
|
|
mDataplex->removeTransferModule(fc->mTransfer->hash());
|
2008-07-23 18:01:59 -04:00
|
|
|
|
2008-10-29 16:58:23 -04:00
|
|
|
if (fc->mTransfer)
|
|
|
|
{
|
|
|
|
delete fc->mTransfer;
|
|
|
|
fc->mTransfer = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fc->mCreator)
|
|
|
|
{
|
|
|
|
delete fc->mCreator;
|
|
|
|
fc->mCreator = NULL;
|
|
|
|
}
|
2008-07-23 18:01:59 -04:00
|
|
|
|
2008-09-11 06:33:43 -04:00
|
|
|
fc->mState = ftFileControl::COMPLETED;
|
2008-07-23 18:01:59 -04:00
|
|
|
|
2008-10-29 16:58:23 -04:00
|
|
|
/* Move to Correct Location */
|
|
|
|
if (0 == rename(fc->mCurrentPath.c_str(), fc->mDestination.c_str()))
|
|
|
|
{
|
2008-11-09 17:17:20 -05:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::completeFile() renaming to: ";
|
|
|
|
std::cerr << fc->mDestination;
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
|
2008-10-29 16:58:23 -04:00
|
|
|
/* correct the file_name */
|
|
|
|
fc->mCurrentPath = fc->mDestination;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2008-11-09 17:17:20 -05:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::completeFile() FAILED mv to: ";
|
|
|
|
std::cerr << fc->mDestination;
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
|
2008-10-29 16:58:23 -04:00
|
|
|
fc->mState = ftFileControl::ERROR_COMPLETION;
|
|
|
|
}
|
|
|
|
|
2008-11-09 17:17:20 -05:00
|
|
|
/* switch map */
|
|
|
|
mCompleted[fc->mHash] = *fc;
|
|
|
|
mDownloads.erase(it);
|
|
|
|
|
|
|
|
|
|
|
|
/* for extralist additions */
|
|
|
|
path = fc->mDestination;
|
|
|
|
//hash = fc->mHash;
|
|
|
|
size = fc->mSize;
|
|
|
|
state = fc->mState;
|
|
|
|
period = 30 * 24 * 3600; /* 30 days */
|
|
|
|
flags = 0;
|
|
|
|
|
|
|
|
doCallback = fc->mDoCallback;
|
|
|
|
callbackCode = fc->mCallbackCode;
|
|
|
|
|
|
|
|
} /******* UNLOCKED ********/
|
|
|
|
|
|
|
|
|
|
|
|
/******************** NO Mutex from Now ********************
|
|
|
|
* cos Callback can end up back in this class.
|
|
|
|
***********************************************************/
|
2008-10-29 16:58:23 -04:00
|
|
|
|
|
|
|
/* If it has a callback - do it now */
|
2008-11-09 17:17:20 -05:00
|
|
|
if (doCallback)
|
2008-10-29 16:58:23 -04:00
|
|
|
{
|
2008-11-04 18:12:53 -05:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::completeFile() doing Callback";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
2008-11-09 17:17:20 -05:00
|
|
|
switch (callbackCode)
|
2008-10-29 16:58:23 -04:00
|
|
|
{
|
|
|
|
case CB_CODE_CACHE:
|
|
|
|
/* callback */
|
2008-11-09 17:17:20 -05:00
|
|
|
if (state == ftFileControl::COMPLETED)
|
2008-10-29 16:58:23 -04:00
|
|
|
{
|
2008-11-04 18:12:53 -05:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::completeFile() doing Callback : Success";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
|
2008-11-09 17:17:20 -05:00
|
|
|
CompletedCache(hash);
|
2008-10-29 16:58:23 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2008-11-04 18:12:53 -05:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::completeFile() Cache Callback : Failed";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
2008-11-09 17:17:20 -05:00
|
|
|
FailedCache(hash);
|
2008-10-29 16:58:23 -04:00
|
|
|
}
|
2008-11-09 17:17:20 -05:00
|
|
|
break;
|
|
|
|
case CB_CODE_EXTRA:
|
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::completeFile() adding to ExtraList";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
mExtraList->addExtraFile(path, hash, size, period, flags);
|
|
|
|
|
|
|
|
|
2008-10-29 16:58:23 -04:00
|
|
|
break;
|
|
|
|
case CB_CODE_MEDIA:
|
2008-11-04 18:12:53 -05:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::completeFile() NULL MEDIA callback";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
2008-10-29 16:58:23 -04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2008-11-04 18:12:53 -05:00
|
|
|
else
|
|
|
|
{
|
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::completeFile() No callback";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
}
|
2008-10-29 16:58:23 -04:00
|
|
|
|
2008-11-15 15:00:29 -05:00
|
|
|
IndicateConfigChanged(); /* completed transfer -> save */
|
2008-07-23 18:01:59 -04:00
|
|
|
return true;
|
2008-08-09 13:03:24 -04:00
|
|
|
|
2008-07-23 18:01:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/***************************************************************/
|
|
|
|
/********************** Controller Access **********************/
|
|
|
|
/***************************************************************/
|
|
|
|
|
2008-11-18 20:12:21 -05:00
|
|
|
const uint32_t FT_CNTRL_STANDARD_RATE = 1024 * 1024;
|
|
|
|
const uint32_t FT_CNTRL_SLOW_RATE = 10 * 1024;
|
2008-11-04 18:12:53 -05:00
|
|
|
|
2008-07-23 18:01:59 -04:00
|
|
|
bool ftController::FileRequest(std::string fname, std::string hash,
|
|
|
|
uint64_t size, std::string dest, uint32_t flags,
|
2008-08-09 13:03:24 -04:00
|
|
|
std::list<std::string> &srcIds)
|
2008-07-23 18:01:59 -04:00
|
|
|
{
|
|
|
|
/* check if we have the file */
|
2008-08-09 13:03:24 -04:00
|
|
|
FileInfo info;
|
2008-08-17 11:23:11 -04:00
|
|
|
std::list<std::string>::iterator it;
|
2008-11-02 06:38:11 -05:00
|
|
|
std::list<TransferInfo>::iterator pit;
|
2008-07-23 18:01:59 -04:00
|
|
|
|
2008-08-17 11:23:11 -04:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::FileRequest(" << fname << ",";
|
|
|
|
std::cerr << hash << "," << size << "," << dest << ",";
|
|
|
|
std::cerr << flags << ",<";
|
|
|
|
|
|
|
|
for(it = srcIds.begin(); it != srcIds.end(); it++)
|
2008-07-23 18:01:59 -04:00
|
|
|
{
|
2008-08-17 11:23:11 -04:00
|
|
|
std::cerr << *it << ",";
|
2008-07-23 18:01:59 -04:00
|
|
|
}
|
2008-08-17 11:23:11 -04:00
|
|
|
std::cerr << ">)";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
|
2008-11-13 18:03:46 -05:00
|
|
|
std::string ownId = mConnMgr->getOwnId();
|
2008-11-09 17:17:20 -05:00
|
|
|
uint32_t rate = 0;
|
|
|
|
if (flags & RS_FILE_HINTS_BACKGROUND)
|
|
|
|
{
|
|
|
|
rate = FT_CNTRL_SLOW_RATE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
rate = FT_CNTRL_STANDARD_RATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* First check if the file is already being downloaded....
|
|
|
|
* This is important as some guis request duplicate files regularly.
|
|
|
|
*/
|
|
|
|
|
2008-11-15 15:00:29 -05:00
|
|
|
{ RsStackMutex stack(ctrlMutex); /******* LOCKED ********/
|
|
|
|
|
2008-11-09 17:17:20 -05:00
|
|
|
std::map<std::string, ftFileControl>::iterator dit;
|
|
|
|
dit = mDownloads.find(hash);
|
|
|
|
if (dit != mDownloads.end())
|
|
|
|
{
|
|
|
|
/* we already have it! */
|
|
|
|
|
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::FileRequest() Already Downloading File";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
std::cerr << "\tNo need to download";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
/* but we should add this peer - if they don't exist!
|
|
|
|
* (needed for channels).
|
|
|
|
*/
|
|
|
|
|
|
|
|
for(it = srcIds.begin(); it != srcIds.end(); it++)
|
|
|
|
{
|
|
|
|
uint32_t i, j;
|
|
|
|
if ((dit->second).mTransfer->getPeerState(*it, i, j))
|
|
|
|
{
|
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::FileRequest() Peer Existing";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
continue; /* already added peer */
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::FileRequest() Adding Peer: " << *it;
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
2008-11-13 18:03:46 -05:00
|
|
|
(dit->second).mTransfer->addFileSource(*it);
|
|
|
|
setPeerState(dit->second.mTransfer, *it,
|
|
|
|
rate, mConnMgr->isOnline(*it));
|
2008-11-15 15:00:29 -05:00
|
|
|
|
|
|
|
IndicateConfigChanged(); /* new peer for transfer -> save */
|
2008-11-13 18:03:46 -05:00
|
|
|
}
|
2008-11-09 17:17:20 -05:00
|
|
|
|
2008-11-13 18:03:46 -05:00
|
|
|
if (srcIds.size() == 0)
|
|
|
|
{
|
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::FileRequest() WARNING: No Src Peers";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
2008-11-09 17:17:20 -05:00
|
|
|
}
|
2008-11-13 18:03:46 -05:00
|
|
|
|
|
|
|
return true;
|
2008-11-09 17:17:20 -05:00
|
|
|
}
|
2008-11-15 15:00:29 -05:00
|
|
|
} /******* UNLOCKED ********/
|
2008-11-09 17:17:20 -05:00
|
|
|
|
2008-10-29 16:58:23 -04:00
|
|
|
bool doCallback = false;
|
|
|
|
uint32_t callbackCode = 0;
|
|
|
|
if (flags & RS_FILE_HINTS_NO_SEARCH)
|
2008-08-17 11:23:11 -04:00
|
|
|
{
|
2008-10-29 16:58:23 -04:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::FileRequest() Flags for NO_SEARCH ";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
2008-08-17 11:23:11 -04:00
|
|
|
/* no search */
|
2008-10-29 16:58:23 -04:00
|
|
|
if (flags & RS_FILE_HINTS_CACHE)
|
|
|
|
{
|
|
|
|
doCallback = true;
|
|
|
|
callbackCode = CB_CODE_CACHE;
|
|
|
|
}
|
2008-11-09 17:17:20 -05:00
|
|
|
else if (flags & RS_FILE_HINTS_EXTRA)
|
|
|
|
{
|
|
|
|
doCallback = true;
|
|
|
|
callbackCode = CB_CODE_EXTRA;
|
|
|
|
}
|
2008-08-17 11:23:11 -04:00
|
|
|
}
|
|
|
|
else
|
2008-07-23 18:01:59 -04:00
|
|
|
{
|
2008-08-17 11:23:11 -04:00
|
|
|
if (mSearch->search(hash, size,
|
|
|
|
RS_FILE_HINTS_LOCAL |
|
|
|
|
RS_FILE_HINTS_EXTRA |
|
|
|
|
RS_FILE_HINTS_SPEC_ONLY, info))
|
|
|
|
{
|
|
|
|
/* have it already */
|
|
|
|
/* add in as completed transfer */
|
2008-10-29 16:58:23 -04:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::FileRequest() Matches Local File";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
std::cerr << "\tNo need to download";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
2008-08-17 11:23:11 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* do a source search - for any extra sources */
|
|
|
|
if (mSearch->search(hash, size,
|
|
|
|
RS_FILE_HINTS_REMOTE |
|
|
|
|
RS_FILE_HINTS_SPEC_ONLY, info))
|
|
|
|
{
|
|
|
|
/* do something with results */
|
2008-10-29 16:58:23 -04:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::FileRequest() Found Other Sources";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* if the sources don't exist already - add in */
|
2008-11-02 06:38:11 -05:00
|
|
|
for(pit = info.peers.begin(); pit != info.peers.end(); pit++)
|
2008-10-29 16:58:23 -04:00
|
|
|
{
|
2008-11-02 06:38:11 -05:00
|
|
|
std::cerr << "\tSource: " << pit->peerId;
|
2008-10-29 16:58:23 -04:00
|
|
|
std::cerr << std::endl;
|
|
|
|
|
|
|
|
if (srcIds.end() == std::find(
|
2008-11-02 06:38:11 -05:00
|
|
|
srcIds.begin(), srcIds.end(), pit->peerId))
|
2008-10-29 16:58:23 -04:00
|
|
|
{
|
2008-11-02 06:38:11 -05:00
|
|
|
srcIds.push_back(pit->peerId);
|
2008-10-29 16:58:23 -04:00
|
|
|
|
2008-11-02 06:38:11 -05:00
|
|
|
std::cerr << "\tAdding in: " << pit->peerId;
|
2008-10-29 16:58:23 -04:00
|
|
|
std::cerr << std::endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-11-09 17:17:20 -05:00
|
|
|
if (flags & RS_FILE_HINTS_EXTRA)
|
|
|
|
{
|
|
|
|
doCallback = true;
|
|
|
|
callbackCode = CB_CODE_EXTRA;
|
|
|
|
}
|
|
|
|
else if (flags & RS_FILE_HINTS_MEDIA)
|
2008-10-29 16:58:23 -04:00
|
|
|
{
|
|
|
|
doCallback = true;
|
|
|
|
callbackCode = CB_CODE_MEDIA;
|
2008-08-17 11:23:11 -04:00
|
|
|
}
|
2008-07-23 18:01:59 -04:00
|
|
|
}
|
|
|
|
|
2008-10-29 16:58:23 -04:00
|
|
|
//std::map<std::string, ftTransferModule *> mTransfers;
|
|
|
|
//std::map<std::string, ftFileCreator *> mFileCreators;
|
2008-07-23 18:01:59 -04:00
|
|
|
|
|
|
|
/* add in new item for download */
|
2008-11-15 15:00:29 -05:00
|
|
|
std::string savepath;
|
|
|
|
std::string destination;
|
|
|
|
|
|
|
|
{ RsStackMutex stack(ctrlMutex); /******* LOCKED ********/
|
|
|
|
|
|
|
|
savepath = mPartialsPath + "/" + hash;
|
|
|
|
destination = dest + "/" + fname;
|
2008-10-29 16:58:23 -04:00
|
|
|
|
|
|
|
/* if no destpath - send to download directory */
|
|
|
|
if (dest == "")
|
|
|
|
{
|
|
|
|
destination = mDownloadPath + "/" + fname;
|
|
|
|
}
|
2008-11-15 15:00:29 -05:00
|
|
|
} /******* UNLOCKED ********/
|
2008-10-29 16:58:23 -04:00
|
|
|
|
2008-10-22 14:12:58 -04:00
|
|
|
ftFileCreator *fc = new ftFileCreator(savepath, size, hash, 0);
|
2008-09-08 10:04:10 -04:00
|
|
|
ftTransferModule *tm = new ftTransferModule(fc, mDataplex,this);
|
2008-07-23 18:01:59 -04:00
|
|
|
|
|
|
|
/* add into maps */
|
2008-10-29 16:58:23 -04:00
|
|
|
ftFileControl ftfc(fname, savepath, destination,
|
|
|
|
size, hash, flags, fc, tm, callbackCode);
|
2008-07-23 18:01:59 -04:00
|
|
|
|
2008-11-13 18:03:46 -05:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::FileRequest() Created ftFileCreator @: " << fc;
|
|
|
|
std::cerr << std::endl;
|
|
|
|
std::cerr << "ftController::FileRequest() Created ftTransModule @: " << tm;
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
|
2008-07-23 18:01:59 -04:00
|
|
|
/* add to ClientModule */
|
2008-08-09 13:03:24 -04:00
|
|
|
mDataplex->addTransferModule(tm, fc);
|
2008-07-23 18:01:59 -04:00
|
|
|
|
|
|
|
/* now add source peers (and their current state) */
|
|
|
|
tm->setFileSources(srcIds);
|
|
|
|
|
|
|
|
/* get current state for transfer module */
|
|
|
|
for(it = srcIds.begin(); it != srcIds.end(); it++)
|
|
|
|
{
|
2008-08-21 17:30:59 -04:00
|
|
|
#ifdef CONTROL_DEBUG
|
2008-11-13 18:03:46 -05:00
|
|
|
std::cerr << "ftController::FileRequest() adding peer: " << *it;
|
|
|
|
std::cerr << std::endl;
|
2008-08-17 11:23:11 -04:00
|
|
|
#endif
|
2008-11-13 18:03:46 -05:00
|
|
|
setPeerState(tm, *it, rate, mConnMgr->isOnline(*it));
|
2008-07-23 18:01:59 -04:00
|
|
|
}
|
|
|
|
|
2008-11-15 15:00:29 -05:00
|
|
|
|
2008-07-23 18:01:59 -04:00
|
|
|
RsStackMutex stack(ctrlMutex); /******* LOCKED ********/
|
|
|
|
mDownloads[hash] = ftfc;
|
|
|
|
mSlowQueue.push_back(hash);
|
|
|
|
|
2008-11-15 15:00:29 -05:00
|
|
|
|
|
|
|
IndicateConfigChanged(); /* completed transfer -> save */
|
2008-10-29 16:58:23 -04:00
|
|
|
return true;
|
2008-08-09 13:03:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-13 18:03:46 -05:00
|
|
|
bool ftController::setPeerState(ftTransferModule *tm, std::string id,
|
|
|
|
uint32_t maxrate, bool online)
|
|
|
|
{
|
|
|
|
if (id == mConnMgr->getOwnId())
|
|
|
|
{
|
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::setPeerState() is Self";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
tm->setPeerState(id, PQIPEER_IDLE, maxrate);
|
|
|
|
}
|
|
|
|
else if (online)
|
|
|
|
{
|
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::setPeerState()";
|
|
|
|
std::cerr << " Peer is Online";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
tm->setPeerState(id, PQIPEER_IDLE, maxrate);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::setPeerState()";
|
|
|
|
std::cerr << " Peer is Offline";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
tm->setPeerState(id, PQIPEER_NOT_ONLINE, maxrate);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-04 18:12:53 -05:00
|
|
|
|
2008-08-09 13:03:24 -04:00
|
|
|
bool ftController::FileCancel(std::string hash)
|
|
|
|
{
|
2008-09-08 04:44:37 -04:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::FileCancel" << std::endl;
|
|
|
|
#endif
|
|
|
|
/*check if the file in the download map*/
|
|
|
|
std::map<std::string,ftFileControl>::iterator mit;
|
|
|
|
mit=mDownloads.find(hash);
|
|
|
|
if (mit==mDownloads.end())
|
|
|
|
{
|
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr<<"ftController::FileCancel file is not found in mDownloads"<<std::endl;
|
|
|
|
#endif
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*find the point to transfer module*/
|
|
|
|
ftTransferModule* ft=(mit->second).mTransfer;
|
|
|
|
ft->cancelTransfer();
|
|
|
|
return true;
|
2008-07-23 18:01:59 -04:00
|
|
|
}
|
|
|
|
|
2008-08-09 13:03:24 -04:00
|
|
|
bool ftController::FileControl(std::string hash, uint32_t flags)
|
|
|
|
{
|
2008-09-08 04:44:37 -04:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::FileControl(" << hash << ",";
|
|
|
|
std::cerr << flags << ")"<<std::endl;
|
|
|
|
#endif
|
|
|
|
/*check if the file in the download map*/
|
|
|
|
std::map<std::string,ftFileControl>::iterator mit;
|
|
|
|
mit=mDownloads.find(hash);
|
|
|
|
if (mit==mDownloads.end())
|
|
|
|
{
|
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr<<"ftController::FileControl file is not found in mDownloads"<<std::endl;
|
|
|
|
#endif
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*find the point to transfer module*/
|
|
|
|
ftTransferModule* ft=(mit->second).mTransfer;
|
|
|
|
switch (flags)
|
|
|
|
{
|
|
|
|
case RS_FILE_CTRL_PAUSE:
|
2008-09-09 09:08:22 -04:00
|
|
|
ft->pauseTransfer();
|
2008-09-08 04:44:37 -04:00
|
|
|
break;
|
|
|
|
case RS_FILE_CTRL_START:
|
|
|
|
ft->resumeTransfer();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
2008-08-09 13:03:24 -04:00
|
|
|
}
|
2008-07-23 18:01:59 -04:00
|
|
|
|
2008-08-09 13:03:24 -04:00
|
|
|
bool ftController::FileClearCompleted()
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2008-07-23 18:01:59 -04:00
|
|
|
|
|
|
|
/* get Details of File Transfers */
|
|
|
|
bool ftController::FileDownloads(std::list<std::string> &hashs)
|
|
|
|
{
|
|
|
|
RsStackMutex stack(ctrlMutex); /******* LOCKED ********/
|
|
|
|
|
|
|
|
std::map<std::string, ftFileControl>::iterator it;
|
|
|
|
for(it = mDownloads.begin(); it != mDownloads.end(); it++)
|
|
|
|
{
|
2008-08-09 13:03:24 -04:00
|
|
|
hashs.push_back(it->second.mHash);
|
2008-07-23 18:01:59 -04:00
|
|
|
}
|
2008-11-09 17:17:20 -05:00
|
|
|
for(it = mCompleted.begin(); it != mCompleted.end(); it++)
|
|
|
|
{
|
|
|
|
hashs.push_back(it->second.mHash);
|
|
|
|
}
|
2008-07-23 18:01:59 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Directory Handling */
|
|
|
|
bool ftController::setDownloadDirectory(std::string path)
|
|
|
|
{
|
2008-08-29 21:07:24 -04:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::setDownloadDirectory(" << path << ")";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
2008-07-23 18:01:59 -04:00
|
|
|
/* check if it exists */
|
|
|
|
if (RsDirUtil::checkCreateDirectory(path))
|
|
|
|
{
|
|
|
|
RsStackMutex stack(ctrlMutex); /******* LOCKED ********/
|
|
|
|
|
|
|
|
mDownloadPath = path;
|
2008-08-29 21:07:24 -04:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::setDownloadDirectory() Okay!";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
2008-07-23 18:01:59 -04:00
|
|
|
return true;
|
|
|
|
}
|
2008-08-29 21:07:24 -04:00
|
|
|
|
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::setDownloadDirectory() Failed";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
2008-07-23 18:01:59 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-08-09 13:03:24 -04:00
|
|
|
bool ftController::setPartialsDirectory(std::string path)
|
2008-07-23 18:01:59 -04:00
|
|
|
{
|
2008-08-09 13:03:24 -04:00
|
|
|
|
2008-07-23 18:01:59 -04:00
|
|
|
/* check it is not a subdir of download / shared directories (BAD) - TODO */
|
|
|
|
|
|
|
|
/* check if it exists */
|
|
|
|
|
|
|
|
if (RsDirUtil::checkCreateDirectory(path))
|
|
|
|
{
|
|
|
|
RsStackMutex stack(ctrlMutex); /******* LOCKED ********/
|
|
|
|
|
2008-10-29 16:58:23 -04:00
|
|
|
mPartialsPath = path;
|
2008-07-23 18:01:59 -04:00
|
|
|
|
2008-10-29 16:58:23 -04:00
|
|
|
#if 0 /*** FIX ME !!!**************/
|
2008-07-23 18:01:59 -04:00
|
|
|
/* move all existing files! */
|
|
|
|
std::map<std::string, ftFileControl>::iterator it;
|
|
|
|
for(it = mDownloads.begin(); it != mDownloads.end(); it++)
|
|
|
|
{
|
|
|
|
(it->second).mCreator->changePartialDirectory(mPartialPath);
|
|
|
|
}
|
2008-10-29 16:58:23 -04:00
|
|
|
#endif
|
2008-07-23 18:01:59 -04:00
|
|
|
return true;
|
|
|
|
}
|
2008-08-09 13:03:24 -04:00
|
|
|
|
2008-07-23 18:01:59 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string ftController::getDownloadDirectory()
|
|
|
|
{
|
|
|
|
RsStackMutex stack(ctrlMutex); /******* LOCKED ********/
|
|
|
|
|
|
|
|
return mDownloadPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string ftController::getPartialsDirectory()
|
|
|
|
{
|
|
|
|
RsStackMutex stack(ctrlMutex); /******* LOCKED ********/
|
|
|
|
|
2008-10-29 16:58:23 -04:00
|
|
|
return mPartialsPath;
|
2008-07-23 18:01:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ftController::FileDetails(std::string hash, FileInfo &info)
|
|
|
|
{
|
|
|
|
RsStackMutex stack(ctrlMutex); /******* LOCKED ********/
|
|
|
|
|
2008-11-09 17:17:20 -05:00
|
|
|
bool completed = false;
|
2008-07-23 18:01:59 -04:00
|
|
|
std::map<std::string, ftFileControl>::iterator it;
|
|
|
|
it = mDownloads.find(hash);
|
|
|
|
if (it == mDownloads.end())
|
|
|
|
{
|
2008-11-09 17:17:20 -05:00
|
|
|
/* search completed files too */
|
|
|
|
it = mCompleted.find(hash);
|
|
|
|
if (it == mCompleted.end())
|
|
|
|
{
|
|
|
|
/* Note: mTransfer & mCreator
|
|
|
|
* are both NULL
|
|
|
|
*/
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
completed = true;
|
2008-07-23 18:01:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* extract details */
|
2008-11-02 06:38:11 -05:00
|
|
|
info.hash = hash;
|
|
|
|
info.fname = it->second.mName;
|
2008-11-15 15:00:29 -05:00
|
|
|
info.path = RsDirUtil::removeTopDir(it->second.mDestination); /* remove fname */
|
2008-11-02 06:38:11 -05:00
|
|
|
|
|
|
|
/* get list of sources from transferModule */
|
|
|
|
std::list<std::string> peerIds;
|
|
|
|
std::list<std::string>::iterator pit;
|
|
|
|
|
2008-11-09 17:17:20 -05:00
|
|
|
if (!completed)
|
|
|
|
{
|
|
|
|
it->second.mTransfer->getFileSources(peerIds);
|
|
|
|
}
|
2008-11-02 06:38:11 -05:00
|
|
|
|
|
|
|
double totalRate;
|
|
|
|
uint32_t tfRate;
|
|
|
|
uint32_t state;
|
|
|
|
|
|
|
|
bool isDownloading = false;
|
|
|
|
bool isSuspended = false;
|
|
|
|
|
|
|
|
for(pit = peerIds.begin(); pit != peerIds.end(); pit++)
|
|
|
|
{
|
|
|
|
if (it->second.mTransfer->getPeerState(*pit, state, tfRate))
|
|
|
|
{
|
|
|
|
TransferInfo ti;
|
|
|
|
switch(state)
|
|
|
|
{
|
|
|
|
case PQIPEER_INIT:
|
|
|
|
ti.status = FT_STATE_OKAY;
|
|
|
|
break;
|
|
|
|
case PQIPEER_NOT_ONLINE:
|
|
|
|
ti.status = FT_STATE_WAITING;
|
|
|
|
break;
|
|
|
|
case PQIPEER_DOWNLOADING:
|
|
|
|
isDownloading = true;
|
|
|
|
ti.status = FT_STATE_DOWNLOADING;
|
|
|
|
break;
|
|
|
|
case PQIPEER_IDLE:
|
|
|
|
ti.status = FT_STATE_OKAY;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
case PQIPEER_SUSPEND:
|
|
|
|
isSuspended = true;
|
|
|
|
ti.status = FT_STATE_FAILED;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ti.tfRate = tfRate / 1024.0;
|
|
|
|
ti.peerId = *pit;
|
|
|
|
info.peers.push_back(ti);
|
|
|
|
totalRate += tfRate / 1024.0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-11-09 17:17:20 -05:00
|
|
|
if ((completed) || ((it->second).mCreator->finished()))
|
2008-11-02 06:38:11 -05:00
|
|
|
{
|
|
|
|
info.downloadStatus = FT_STATE_COMPLETE;
|
|
|
|
}
|
|
|
|
else if (isDownloading)
|
|
|
|
{
|
|
|
|
info.downloadStatus = FT_STATE_DOWNLOADING;
|
|
|
|
}
|
|
|
|
else if (isSuspended)
|
|
|
|
{
|
|
|
|
info.downloadStatus = FT_STATE_FAILED;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
info.downloadStatus = FT_STATE_WAITING;
|
|
|
|
}
|
|
|
|
info.tfRate = totalRate;
|
|
|
|
info.size = (it->second).mSize;
|
2008-11-09 17:17:20 -05:00
|
|
|
|
|
|
|
if (completed)
|
|
|
|
{
|
|
|
|
info.transfered = info.size;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
info.transfered = (it->second).mCreator->getRecvd();
|
|
|
|
}
|
2008-07-23 18:01:59 -04:00
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/***************************************************************/
|
|
|
|
/********************** Controller Access **********************/
|
|
|
|
/***************************************************************/
|
|
|
|
|
|
|
|
/* pqiMonitor callback:
|
|
|
|
* Used to tell TransferModules new available peers
|
|
|
|
*/
|
|
|
|
void ftController::statusChange(const std::list<pqipeer> &plist)
|
|
|
|
{
|
|
|
|
RsStackMutex stack(ctrlMutex); /******* LOCKED ********/
|
2008-11-13 18:03:46 -05:00
|
|
|
uint32_t rate = FT_CNTRL_STANDARD_RATE;
|
2008-07-23 18:01:59 -04:00
|
|
|
|
|
|
|
/* add online to all downloads */
|
|
|
|
std::map<std::string, ftFileControl>::iterator it;
|
|
|
|
std::list<pqipeer>::const_iterator pit;
|
|
|
|
|
2008-11-13 18:03:46 -05:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::statusChange()";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
|
2008-07-23 18:01:59 -04:00
|
|
|
for(it = mDownloads.begin(); it != mDownloads.end(); it++)
|
|
|
|
{
|
2008-11-13 18:03:46 -05:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::statusChange() Updating Hash:";
|
|
|
|
std::cerr << it->first;
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
2008-07-23 18:01:59 -04:00
|
|
|
for(pit = plist.begin(); pit != plist.end(); pit++)
|
|
|
|
{
|
2008-11-13 18:03:46 -05:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "Peer: " << pit->id;
|
|
|
|
#endif
|
|
|
|
if (pit->actions & RS_PEER_CONNECTED)
|
|
|
|
{
|
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << " is Newly Connected!";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
setPeerState(it->second.mTransfer, pit->id, rate, true);
|
|
|
|
}
|
|
|
|
else if (pit->actions & RS_PEER_DISCONNECTED)
|
2008-07-23 18:01:59 -04:00
|
|
|
{
|
2008-11-13 18:03:46 -05:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << " is Just disconnected!";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
setPeerState(it->second.mTransfer, pit->id, rate, false);
|
2008-07-23 18:01:59 -04:00
|
|
|
}
|
2008-11-13 18:03:46 -05:00
|
|
|
else
|
2008-07-23 18:01:59 -04:00
|
|
|
{
|
2008-11-13 18:03:46 -05:00
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << " had something happen to it: ";
|
|
|
|
std::cerr << pit-> actions;
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
setPeerState(it->second.mTransfer, pit->id, rate, false);
|
2008-07-23 18:01:59 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2008-08-09 13:03:24 -04:00
|
|
|
}
|
2008-07-23 18:01:59 -04:00
|
|
|
|
2008-08-17 11:23:11 -04:00
|
|
|
/* Cache Interface */
|
|
|
|
bool ftController::RequestCacheFile(RsPeerId id, std::string path, std::string hash, uint64_t size)
|
|
|
|
{
|
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::RequestCacheFile(" << id << ",";
|
|
|
|
std::cerr << path << "," << hash << "," << size << ")";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Request File */
|
|
|
|
std::list<std::string> ids;
|
|
|
|
ids.push_back(id);
|
|
|
|
|
|
|
|
FileRequest(hash, hash, size, path,
|
|
|
|
RS_FILE_HINTS_CACHE | RS_FILE_HINTS_NO_SEARCH, ids);
|
2008-10-29 16:58:23 -04:00
|
|
|
|
|
|
|
return true;
|
2008-08-17 11:23:11 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool ftController::CancelCacheFile(RsPeerId id, std::string path, std::string hash, uint64_t size)
|
|
|
|
{
|
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::CancelCacheFile(" << id << ",";
|
|
|
|
std::cerr << path << "," << hash << "," << size << ")";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
|
2008-10-29 16:58:23 -04:00
|
|
|
return true;
|
2008-08-17 11:23:11 -04:00
|
|
|
}
|
|
|
|
|
2008-11-15 15:00:29 -05:00
|
|
|
const std::string download_dir_ss("DOWN_DIR");
|
|
|
|
const std::string partial_dir_ss("PART_DIR");
|
|
|
|
|
|
|
|
|
|
|
|
/* p3Config Interface */
|
|
|
|
RsSerialiser *ftController::setupSerialiser()
|
|
|
|
{
|
|
|
|
RsSerialiser *rss = new RsSerialiser();
|
|
|
|
|
|
|
|
/* add in the types we need! */
|
|
|
|
rss->addSerialType(new RsFileConfigSerialiser());
|
|
|
|
rss->addSerialType(new RsGeneralConfigSerialiser());
|
|
|
|
|
|
|
|
return rss;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
std::list<RsItem *> ftController::saveList(bool &cleanup)
|
|
|
|
{
|
|
|
|
std::list<RsItem *> saveData;
|
|
|
|
|
|
|
|
/* it can delete them! */
|
|
|
|
cleanup = true;
|
|
|
|
|
|
|
|
/* create a key/value set for most of the parameters */
|
|
|
|
std::map<std::string, std::string> configMap;
|
|
|
|
std::map<std::string, std::string>::iterator mit;
|
|
|
|
std::list<std::string>::iterator it;
|
|
|
|
|
|
|
|
/* basic control parameters */
|
|
|
|
configMap[download_dir_ss] = getDownloadDirectory();
|
|
|
|
configMap[partial_dir_ss] = getPartialsDirectory();
|
|
|
|
|
|
|
|
RsConfigKeyValueSet *rskv = new RsConfigKeyValueSet();
|
|
|
|
|
|
|
|
/* Convert to TLV */
|
|
|
|
for(mit = configMap.begin(); mit != configMap.end(); mit++)
|
|
|
|
{
|
|
|
|
RsTlvKeyValue kv;
|
|
|
|
kv.key = mit->first;
|
|
|
|
kv.value = mit->second;
|
|
|
|
|
|
|
|
rskv->tlvkvs.pairs.push_back(kv);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add KeyValue to saveList */
|
|
|
|
saveData.push_back(rskv);
|
|
|
|
|
|
|
|
/* get list of Downloads ....
|
|
|
|
* strip out Caches / ExtraList / Channels????
|
|
|
|
* (anything with a callback?)
|
|
|
|
* - most systems will restart missing files.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
/* get Details of File Transfers */
|
|
|
|
std::list<std::string> hashs;
|
|
|
|
FileDownloads(hashs);
|
|
|
|
|
|
|
|
for(it = hashs.begin(); it != hashs.end(); it++)
|
|
|
|
{
|
|
|
|
/* stack mutex released each loop */
|
|
|
|
RsStackMutex stack(ctrlMutex); /******* LOCKED ********/
|
|
|
|
|
|
|
|
std::map<std::string, ftFileControl>::iterator fit;
|
|
|
|
fit = mDownloads.find(*it);
|
|
|
|
if (fit == mDownloads.end())
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ignore callback ones */
|
|
|
|
if (fit->second.mDoCallback)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((fit->second).mCreator->finished())
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* make RsFileTransfer item for save list */
|
|
|
|
RsFileTransfer *rft = new RsFileTransfer();
|
|
|
|
|
|
|
|
/* what data is important? */
|
|
|
|
|
|
|
|
rft->file.name = fit->second.mName;
|
|
|
|
rft->file.hash = fit->second.mHash;
|
|
|
|
rft->file.filesize = fit->second.mSize;
|
|
|
|
rft->file.path = RsDirUtil::removeTopDir(fit->second.mDestination); /* remove fname */
|
|
|
|
//rft->flags = fit->second.mFlags;
|
|
|
|
|
|
|
|
fit->second.mTransfer->getFileSources(rft->allPeerIds.ids);
|
|
|
|
|
|
|
|
saveData.push_back(rft);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* list completed! */
|
|
|
|
return saveData;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool ftController::loadList(std::list<RsItem *> load)
|
|
|
|
{
|
|
|
|
std::list<RsItem *>::iterator it;
|
|
|
|
std::list<RsTlvKeyValue>::iterator kit;
|
|
|
|
RsConfigKeyValueSet *rskv;
|
|
|
|
RsFileTransfer *rsft;
|
|
|
|
|
|
|
|
#ifdef CONTROL_DEBUG
|
|
|
|
std::cerr << "ftController::loadList() Item Count: " << load.size();
|
|
|
|
std::cerr << std::endl;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
for(it = load.begin(); it != load.end(); it++)
|
|
|
|
{
|
|
|
|
/* switch on type */
|
|
|
|
if (NULL != (rskv = dynamic_cast<RsConfigKeyValueSet *>(*it)))
|
|
|
|
{
|
|
|
|
/* make into map */
|
|
|
|
std::map<std::string, std::string> configMap;
|
|
|
|
for(kit = rskv->tlvkvs.pairs.begin();
|
|
|
|
kit != rskv->tlvkvs.pairs.end(); kit++)
|
|
|
|
{
|
|
|
|
configMap[kit->key] = kit->value;
|
|
|
|
}
|
|
|
|
|
|
|
|
loadConfigMap(configMap);
|
|
|
|
/* cleanup */
|
|
|
|
delete (*it);
|
|
|
|
|
|
|
|
}
|
|
|
|
else if (NULL != (rsft = dynamic_cast<RsFileTransfer *>(*it)))
|
|
|
|
{
|
|
|
|
RsStackMutex stack(ctrlMutex); /******* LOCKED ********/
|
|
|
|
|
|
|
|
/* save to the preLoad list */
|
|
|
|
mResumeTransferList.push_back(rsft);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* cleanup */
|
|
|
|
delete (*it);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ftController::loadConfigMap(std::map<std::string, std::string> &configMap)
|
|
|
|
{
|
|
|
|
std::map<std::string, std::string>::iterator mit;
|
|
|
|
|
|
|
|
std::string str_true("true");
|
|
|
|
std::string empty("");
|
|
|
|
std::string dir = "notempty";
|
|
|
|
|
|
|
|
if (configMap.end() != (mit = configMap.find(download_dir_ss)))
|
|
|
|
{
|
|
|
|
setDownloadDirectory(mit->second);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (configMap.end() != (mit = configMap.find(partial_dir_ss)))
|
|
|
|
{
|
|
|
|
//setPartialsDirectory(mit->second);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool ftController::ResumeTransfers()
|
|
|
|
{
|
|
|
|
std::list<RsFileTransfer *> resumeList;
|
|
|
|
std::list<RsFileTransfer *>::iterator it;
|
|
|
|
|
|
|
|
{ RsStackMutex stack(ctrlMutex); /******* LOCKED ********/
|
|
|
|
resumeList = mResumeTransferList;
|
|
|
|
mResumeTransferList.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
for(it = resumeList.begin(); it != resumeList.end(); it++)
|
|
|
|
{
|
|
|
|
/* do File request */
|
|
|
|
std::string fname = (*it)->file.name;
|
|
|
|
std::string hash = (*it)->file.hash;
|
|
|
|
uint64_t size = (*it)->file.filesize;
|
|
|
|
std::string dest = (*it)->file.path;
|
|
|
|
uint32_t flags = 0; //(*it)->flags;
|
|
|
|
std::list<std::string> srcIds = (*it)->allPeerIds.ids;
|
|
|
|
|
|
|
|
FileRequest(fname,hash,size,dest,flags,srcIds);
|
|
|
|
|
|
|
|
delete (*it);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2008-08-17 11:23:11 -04:00
|
|
|
|