2015-03-18 14:48:43 -04:00
|
|
|
#include "RsControlModule.h"
|
|
|
|
|
|
|
|
#include <sstream>
|
|
|
|
#include <unistd.h>
|
2015-06-16 11:15:36 -04:00
|
|
|
#include <cstdio>
|
2015-03-18 14:48:43 -04:00
|
|
|
|
|
|
|
#include <retroshare/rsinit.h>
|
|
|
|
#include <retroshare/rsiface.h>
|
|
|
|
|
|
|
|
#include "api/ApiServer.h"
|
|
|
|
#include "api/Operators.h"
|
|
|
|
#include "api/StateTokenServer.h"
|
|
|
|
|
|
|
|
#include "GetPluginInterfaces.h"
|
|
|
|
|
|
|
|
namespace resource_api{
|
|
|
|
|
2015-03-25 04:31:19 -04:00
|
|
|
RsControlModule::RsControlModule(int argc, char **argv, StateTokenServer* sts, ApiServer *apiserver, bool full_control):
|
2015-03-18 14:48:43 -04:00
|
|
|
mStateTokenServer(sts),
|
|
|
|
mApiServer(apiserver),
|
|
|
|
mExitFlagMtx("RsControlModule::mExitFlagMtx"),
|
|
|
|
mProcessShouldExit(false),
|
|
|
|
mDataMtx("RsControlModule::mDataMtx"),
|
|
|
|
mRunState(WAITING_INIT),
|
|
|
|
mAutoLoginNextTime(false),
|
|
|
|
mWantPassword(false)
|
|
|
|
{
|
|
|
|
mStateToken = sts->getNewToken();
|
|
|
|
this->argc = argc;
|
|
|
|
this->argv = argv;
|
|
|
|
// start worker thread
|
2015-03-25 04:31:19 -04:00
|
|
|
if(full_control)
|
2016-06-01 10:18:19 -04:00
|
|
|
start("resapi ctrl mod");
|
2015-03-25 04:31:19 -04:00
|
|
|
else
|
2015-04-29 08:20:52 -04:00
|
|
|
mRunState = RUNNING_OK_NO_FULL_CONTROL;
|
2015-03-18 14:48:43 -04:00
|
|
|
|
|
|
|
addResourceHandler("runstate", this, &RsControlModule::handleRunState);
|
|
|
|
addResourceHandler("identities", this, &RsControlModule::handleIdentities);
|
|
|
|
addResourceHandler("locations", this, &RsControlModule::handleLocations);
|
|
|
|
addResourceHandler("password", this, &RsControlModule::handlePassword);
|
|
|
|
addResourceHandler("login", this, &RsControlModule::handleLogin);
|
|
|
|
addResourceHandler("shutdown", this, &RsControlModule::handleShutdown);
|
2015-06-16 08:35:07 -04:00
|
|
|
addResourceHandler("import_pgp", this, &RsControlModule::handleImportPgp);
|
|
|
|
addResourceHandler("create_location", this, &RsControlModule::handleCreateLocation);
|
2015-03-18 14:48:43 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
RsControlModule::~RsControlModule()
|
|
|
|
{
|
2015-05-22 16:54:38 -04:00
|
|
|
// join();
|
2015-03-18 14:48:43 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool RsControlModule::processShouldExit()
|
|
|
|
{
|
|
|
|
RsStackMutex stack(mExitFlagMtx);
|
|
|
|
return mProcessShouldExit;
|
|
|
|
}
|
|
|
|
|
2016-08-08 19:22:14 -04:00
|
|
|
bool RsControlModule::askForPassword(const std::string &title, const std::string &key_details, bool /* prev_is_bad */, std::string &password, bool& cancelled)
|
2015-03-18 14:48:43 -04:00
|
|
|
{
|
2015-06-11 16:42:01 -04:00
|
|
|
cancelled = false ;
|
2015-03-18 14:48:43 -04:00
|
|
|
{
|
2016-08-02 18:48:19 -04:00
|
|
|
RS_STACK_MUTEX(mDataMtx); // ********** LOCKED **********
|
2015-06-16 08:35:07 -04:00
|
|
|
if(mFixedPassword != "")
|
|
|
|
{
|
|
|
|
password = mFixedPassword;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-03-18 14:48:43 -04:00
|
|
|
mWantPassword = true;
|
2016-08-08 19:22:14 -04:00
|
|
|
mTitle = title;
|
2015-03-18 14:48:43 -04:00
|
|
|
mKeyName = key_details;
|
|
|
|
mPassword = "";
|
|
|
|
mStateTokenServer->replaceToken(mStateToken);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool wait = true;
|
|
|
|
while(wait)
|
|
|
|
{
|
|
|
|
usleep(5*1000);
|
|
|
|
|
2016-08-02 18:48:19 -04:00
|
|
|
RS_STACK_MUTEX(mDataMtx); // ********** LOCKED **********
|
2015-03-18 14:48:43 -04:00
|
|
|
wait = mWantPassword;
|
|
|
|
if(!wait && mPassword != "")
|
|
|
|
{
|
|
|
|
password = mPassword;
|
|
|
|
mPassword = "";
|
|
|
|
mWantPassword = false;
|
|
|
|
mStateTokenServer->replaceToken(mStateToken);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RsControlModule::run()
|
|
|
|
{
|
|
|
|
std::cerr << "RsControlModule: initialising libretroshare..." << std::endl;
|
|
|
|
|
|
|
|
RsInit::InitRsConfig();
|
|
|
|
int initResult = RsInit::InitRetroShare(argc, argv, true);
|
|
|
|
|
|
|
|
if (initResult < 0) {
|
|
|
|
std::cerr << "RsControlModule: FATAL ERROR, initialising libretroshare FAILED." << std::endl;
|
|
|
|
/* Error occured */
|
|
|
|
std::stringstream ss;
|
|
|
|
switch (initResult) {
|
|
|
|
case RS_INIT_AUTH_FAILED:
|
|
|
|
ss << "RsInit::InitRetroShare AuthGPG::InitAuth failed" << std::endl;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* Unexpected return code */
|
|
|
|
ss << "RsInit::InitRetroShare unexpected return code " << initResult << std::endl;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// FATAL ERROR, we can't recover from this. Just send the message to the user.
|
|
|
|
setRunState(FATAL_ERROR, ss.str());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is needed to allocate rsNotify, so that it can be used to ask for PGP passphrase
|
|
|
|
RsControl::earlyInitNotificationSystem();
|
|
|
|
rsNotify->registerNotifyClient(this);
|
|
|
|
|
|
|
|
bool login_ok = false;
|
|
|
|
while(!login_ok)
|
|
|
|
{
|
|
|
|
// skip account selection if autologin is available
|
|
|
|
if(initResult != RS_INIT_HAVE_ACCOUNT)
|
|
|
|
setRunState(WAITING_ACCOUNT_SELECT);
|
|
|
|
|
|
|
|
// wait for login request
|
|
|
|
bool auto_login = false;
|
|
|
|
bool wait_for_account_select = (initResult != RS_INIT_HAVE_ACCOUNT);
|
|
|
|
while(wait_for_account_select && !processShouldExit())
|
|
|
|
{
|
|
|
|
usleep(5*1000);
|
|
|
|
RsStackMutex stack(mDataMtx); // ********** LOCKED **********
|
|
|
|
wait_for_account_select = mLoadPeerId.isNull();
|
|
|
|
auto_login = mAutoLoginNextTime;
|
|
|
|
if(!wait_for_account_select)
|
|
|
|
{
|
|
|
|
wait_for_account_select = !RsAccounts::SelectAccount(mLoadPeerId);
|
|
|
|
if(wait_for_account_select)
|
|
|
|
setRunState(WAITING_ACCOUNT_SELECT);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(processShouldExit())
|
|
|
|
return;
|
|
|
|
|
|
|
|
bool autoLogin = (initResult == RS_INIT_HAVE_ACCOUNT) | auto_login;
|
|
|
|
std::string lockFile;
|
|
|
|
int retVal = RsInit::LockAndLoadCertificates(autoLogin, lockFile);
|
|
|
|
|
|
|
|
std::string error_string;
|
|
|
|
switch (retVal) {
|
|
|
|
case 0:
|
|
|
|
login_ok = true;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
error_string = "Another RetroShare using the same profile is "
|
|
|
|
"already running on your system. Please close "
|
|
|
|
"that instance first\n Lock file:\n" + lockFile;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
error_string = "An unexpected error occurred when Retroshare "
|
|
|
|
"tried to acquire the single instance lock\n Lock file:\n"
|
|
|
|
+ lockFile;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
error_string = "Login Failure: Maybe password is wrong";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
std::cerr << "RsControlModule::run() LockAndLoadCertificates failed. Unexpected switch value: " << retVal << std::endl;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
setRunState(WAITING_STARTUP);
|
|
|
|
|
|
|
|
std::cerr << "RsControlModule: login ok, starting Retroshare worker threads..." << std::endl;
|
|
|
|
RsControl::instance() -> StartupRetroShare();
|
|
|
|
|
|
|
|
std::cerr << "RsControlModule: loading main resource api modules..." << std::endl;
|
|
|
|
RsPlugInInterfaces ifaces;
|
|
|
|
getPluginInterfaces(ifaces);
|
|
|
|
mApiServer->loadMainModules(ifaces);
|
|
|
|
|
|
|
|
std::cerr << "RsControlModule: Retroshare is up and running. Enjoy!" << std::endl;
|
|
|
|
setRunState(RUNNING_OK);
|
|
|
|
|
|
|
|
while(!processShouldExit())
|
|
|
|
{
|
|
|
|
usleep(5*1000);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::cerr << "RsControlModule: stopping Retroshare..." << std::endl;
|
|
|
|
RsControl::instance() -> rsGlobalShutDown();
|
|
|
|
std::cerr << "RsControlModule: Retroshare stopped. Bye!" << std::endl;
|
|
|
|
}
|
|
|
|
|
2015-06-16 08:35:07 -04:00
|
|
|
void RsControlModule::handleRunState(Request &, Response &resp)
|
2015-03-18 14:48:43 -04:00
|
|
|
{
|
|
|
|
RsStackMutex stack(mDataMtx); // ********** LOCKED **********
|
|
|
|
std::string state;
|
|
|
|
switch(mRunState)
|
|
|
|
{
|
|
|
|
case WAITING_INIT:
|
|
|
|
state = "waiting_init";
|
|
|
|
break;
|
|
|
|
case FATAL_ERROR:
|
|
|
|
state = "fatal_error";
|
|
|
|
break;
|
|
|
|
case WAITING_ACCOUNT_SELECT:
|
|
|
|
state = "waiting_account_select";
|
|
|
|
break;
|
|
|
|
case WAITING_STARTUP:
|
|
|
|
state = "waiting_startup";
|
|
|
|
break;
|
|
|
|
case RUNNING_OK:
|
|
|
|
state = "running_ok";
|
|
|
|
break;
|
2015-04-29 08:20:52 -04:00
|
|
|
case RUNNING_OK_NO_FULL_CONTROL:
|
|
|
|
state = "running_ok_no_full_control";
|
|
|
|
break;
|
2015-03-18 14:48:43 -04:00
|
|
|
default:
|
|
|
|
state = "error_should_not_happen_this_is_a_bug";
|
|
|
|
}
|
|
|
|
resp.mDataStream << makeKeyValueReference("runstate", state);
|
|
|
|
resp.mStateToken = mStateToken;
|
|
|
|
resp.setOk();
|
|
|
|
}
|
|
|
|
|
2015-06-16 08:35:07 -04:00
|
|
|
void RsControlModule::handleIdentities(Request &, Response &resp)
|
2015-03-18 14:48:43 -04:00
|
|
|
{
|
|
|
|
RsStackMutex stack(mDataMtx); // ********** LOCKED **********
|
|
|
|
if(mRunState == WAITING_INIT || mRunState == FATAL_ERROR)
|
|
|
|
{
|
|
|
|
resp.setFail("Retroshare is not initialised. Operation not possible.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::list<RsPgpId> pgp_ids;
|
|
|
|
RsAccounts::GetPGPLogins(pgp_ids);
|
|
|
|
resp.mDataStream.getStreamToMember();
|
|
|
|
for(std::list<RsPgpId>::iterator lit = pgp_ids.begin(); lit != pgp_ids.end(); ++lit)
|
|
|
|
{
|
|
|
|
std::string name;
|
|
|
|
std::string email;
|
|
|
|
if(RsAccounts::GetPGPLoginDetails(*lit, name, email))
|
|
|
|
resp.mDataStream.getStreamToMember()
|
|
|
|
<< makeKeyValueReference("id", *lit)
|
|
|
|
<< makeKeyValueReference("pgp_id", *lit)
|
|
|
|
<< makeKeyValueReference("name", name);
|
|
|
|
}
|
|
|
|
resp.mStateToken = mStateToken;
|
|
|
|
resp.setOk();
|
|
|
|
}
|
|
|
|
|
2015-06-16 08:35:07 -04:00
|
|
|
void RsControlModule::handleLocations(Request &, Response &resp)
|
2015-03-18 14:48:43 -04:00
|
|
|
{
|
|
|
|
RsStackMutex stack(mDataMtx); // ********** LOCKED **********
|
|
|
|
if(mRunState == WAITING_INIT || mRunState == FATAL_ERROR)
|
|
|
|
{
|
|
|
|
resp.setFail("Retroshare is not initialised. Operation not possible.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
RsPeerId preferedId;
|
|
|
|
RsAccounts::GetPreferredAccountId(preferedId);
|
|
|
|
|
|
|
|
std::list<RsPeerId> peer_ids;
|
|
|
|
RsAccounts::GetAccountIds(peer_ids);
|
|
|
|
resp.mDataStream.getStreamToMember();
|
|
|
|
for(std::list<RsPeerId>::iterator lit = peer_ids.begin(); lit != peer_ids.end(); ++lit)
|
|
|
|
{
|
|
|
|
bool preferred = preferedId==*lit;
|
|
|
|
RsPgpId pgp_id;
|
|
|
|
std::string pgp_name, pgp_mail, location_name;
|
|
|
|
if(RsAccounts::GetAccountDetails(*lit, pgp_id, pgp_name, pgp_mail, location_name))
|
|
|
|
resp.mDataStream.getStreamToMember()
|
|
|
|
<< makeKeyValueReference("id", *lit)
|
|
|
|
<< makeKeyValueReference("pgp_id", pgp_id)
|
|
|
|
<< makeKeyValueReference("peer_id", *lit)
|
|
|
|
<< makeKeyValueReference("name", pgp_name)
|
|
|
|
<< makeKeyValueReference("location", location_name)
|
|
|
|
<< makeKeyValueReference("preferred", preferred);
|
|
|
|
}
|
|
|
|
resp.mStateToken = mStateToken;
|
|
|
|
resp.setOk();
|
|
|
|
}
|
|
|
|
|
|
|
|
void RsControlModule::handlePassword(Request &req, Response &resp)
|
|
|
|
{
|
|
|
|
RsStackMutex stack(mDataMtx); // ********** LOCKED **********
|
|
|
|
std::string passwd;
|
|
|
|
req.mStream << makeKeyValueReference("password", passwd);
|
|
|
|
if(passwd != "" && mWantPassword)
|
|
|
|
{
|
|
|
|
// client sends password
|
|
|
|
mPassword = passwd;
|
|
|
|
mWantPassword = false;
|
|
|
|
mStateTokenServer->replaceToken(mStateToken);
|
|
|
|
}
|
|
|
|
|
|
|
|
resp.mDataStream
|
|
|
|
<< makeKeyValueReference("want_password", mWantPassword)
|
|
|
|
<< makeKeyValueReference("key_name", mKeyName);
|
|
|
|
resp.mStateToken = mStateToken;
|
|
|
|
resp.setOk();
|
|
|
|
}
|
|
|
|
|
|
|
|
void RsControlModule::handleLogin(Request &req, Response &resp)
|
|
|
|
{
|
|
|
|
RsStackMutex stack(mDataMtx); // ********** LOCKED **********
|
|
|
|
if(mRunState != WAITING_ACCOUNT_SELECT)
|
|
|
|
{
|
|
|
|
resp.setFail("Operation not allowed in this runstate. Login is only allowed rigth after initialisation.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
req.mStream << makeKeyValueReference("id", mLoadPeerId)
|
|
|
|
<< makeKeyValueReference("autologin", mAutoLoginNextTime);
|
|
|
|
resp.setOk();
|
|
|
|
}
|
|
|
|
|
2015-06-16 08:35:07 -04:00
|
|
|
void RsControlModule::handleShutdown(Request &, Response &resp)
|
2015-03-18 14:48:43 -04:00
|
|
|
{
|
|
|
|
RsStackMutex stack(mExitFlagMtx); // ********** LOCKED **********
|
|
|
|
mProcessShouldExit = true;
|
|
|
|
resp.setOk();
|
|
|
|
}
|
|
|
|
|
2015-06-16 08:35:07 -04:00
|
|
|
void RsControlModule::handleImportPgp(Request &req, Response &resp)
|
|
|
|
{
|
|
|
|
std::string key_string;
|
|
|
|
req.mStream << makeKeyValueReference("key_string", key_string);
|
|
|
|
|
|
|
|
if(key_string.empty())
|
|
|
|
{
|
|
|
|
resp.setFail("required field key_string is empty");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
RsPgpId pgp_id;
|
|
|
|
std::string error_string;
|
|
|
|
if(RsAccounts::ImportIdentityFromString(key_string, pgp_id, error_string))
|
|
|
|
{
|
|
|
|
resp.mDataStream << makeKeyValueReference("pgp_id", pgp_id);
|
|
|
|
resp.setOk();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
resp.setFail("Failed to import key: " + error_string);
|
|
|
|
}
|
|
|
|
|
|
|
|
void RsControlModule::handleCreateLocation(Request &req, Response &resp)
|
|
|
|
{
|
|
|
|
std::string hidden_address;
|
|
|
|
std::string hidden_port_str;
|
|
|
|
req.mStream << makeKeyValueReference("hidden_adress", hidden_address)
|
|
|
|
<< makeKeyValueReference("hidden_port", hidden_port_str);
|
|
|
|
uint16_t hidden_port = 0;
|
|
|
|
if(hidden_address.empty() != hidden_port_str.empty())
|
|
|
|
{
|
|
|
|
resp.setFail("you must both specify string hidden_adress and string hidden_port to create a hidden node.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if(!hidden_address.empty())
|
|
|
|
{
|
|
|
|
int p;
|
|
|
|
if(sscanf(hidden_port_str.c_str(), "%i", &p) != 1)
|
|
|
|
{
|
|
|
|
resp.setFail("failed to parse hidden_port, not a number. Must be a dec or hex number.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if(p < 0 || p > 0xFFFF)
|
|
|
|
{
|
|
|
|
resp.setFail("hidden_port out of range. It must fit into uint16!");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
hidden_port = p;
|
|
|
|
}
|
|
|
|
|
|
|
|
RsPgpId pgp_id;
|
|
|
|
std::string pgp_name;
|
|
|
|
std::string pgp_password;
|
|
|
|
std::string ssl_name;
|
|
|
|
|
|
|
|
req.mStream << makeKeyValueReference("pgp_id", pgp_id)
|
|
|
|
<< makeKeyValueReference("pgp_name", pgp_name)
|
|
|
|
<< makeKeyValueReference("pgp_password", pgp_password)
|
|
|
|
<< makeKeyValueReference("ssl_name", ssl_name);
|
|
|
|
|
|
|
|
if(pgp_password.empty())
|
|
|
|
{
|
|
|
|
resp.setFail("Error: pgp_password is empty.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// pgp_id is set: use existing pgp key
|
|
|
|
// pgp_name is set: attempt to create a new key
|
|
|
|
if(pgp_id.isNull() && (pgp_name.empty()||pgp_password.empty()))
|
|
|
|
{
|
|
|
|
resp.setFail("You have to set pgp_id to use an existing key, or pgp_name and pgp_password to create a new pgp key.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if(pgp_id.isNull())
|
|
|
|
{
|
|
|
|
std::string err_string;
|
|
|
|
if(!RsAccounts::GeneratePGPCertificate(pgp_name, "", pgp_password, pgp_id, 2048, err_string))
|
|
|
|
{
|
2015-11-24 08:43:09 -05:00
|
|
|
resp.setFail("could not create pgp key: "+err_string);
|
2015-06-16 08:35:07 -04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(hidden_port)
|
|
|
|
RsInit::SetHiddenLocation(hidden_address, hidden_port);
|
|
|
|
|
|
|
|
std::string ssl_password = RSRandom::random_alphaNumericString(RsInit::getSslPwdLen()) ;
|
|
|
|
|
|
|
|
/* GenerateSSLCertificate - selects the PGP Account */
|
|
|
|
//RsInit::SelectGPGAccount(PGPId);
|
|
|
|
|
|
|
|
RsPeerId ssl_id;
|
|
|
|
std::string err_string;
|
|
|
|
// give the password to the password callback
|
|
|
|
{
|
|
|
|
RsStackMutex stack(mDataMtx); // ********** LOCKED **********
|
|
|
|
mFixedPassword = pgp_password;
|
|
|
|
}
|
|
|
|
bool ssl_ok = RsAccounts::GenerateSSLCertificate(pgp_id, "", ssl_name, "", hidden_port!=0, ssl_password, ssl_id, err_string);
|
|
|
|
|
|
|
|
// clear fixed password to restore normal password operation
|
|
|
|
{
|
|
|
|
RsStackMutex stack(mDataMtx); // ********** LOCKED **********
|
|
|
|
mFixedPassword = "";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ssl_ok)
|
|
|
|
{
|
|
|
|
// load ssl password and load account
|
|
|
|
RsInit::LoadPassword(ssl_password);
|
|
|
|
// trigger login in init thread
|
|
|
|
{
|
|
|
|
RsStackMutex stack(mDataMtx); // ********** LOCKED **********
|
|
|
|
mLoadPeerId = ssl_id;
|
|
|
|
}
|
|
|
|
resp.mDataStream << makeKeyValueReference("pgp_id", pgp_id)
|
|
|
|
<< makeKeyValueReference("pgp_name", pgp_name)
|
|
|
|
<< makeKeyValueReference("ssl_name", ssl_name)
|
|
|
|
<< makeKeyValueReference("ssl_id", ssl_id);
|
|
|
|
resp.setOk();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
resp.setFail("could not create a new location. Error: "+err_string);
|
|
|
|
}
|
|
|
|
|
2015-03-18 14:48:43 -04:00
|
|
|
void RsControlModule::setRunState(RunState s, std::string errstr)
|
|
|
|
{
|
|
|
|
RsStackMutex stack(mDataMtx); // ********** LOCKED **********
|
|
|
|
mRunState = s;
|
|
|
|
mLastErrorString = errstr;
|
|
|
|
mStateTokenServer->replaceToken(mStateToken);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
} // namespace resource_api
|