khanat-opennel-code/code/ryzom/server/src/backup_service/backup_service.cpp
2012-10-05 14:57:05 +02:00

864 lines
23 KiB
C++

// Ryzom - MMORPG Framework <http://dev.ryzom.com/projects/ryzom/>
// Copyright (C) 2010 Winch Gate Property Limited
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero 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 Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
#include "nel/misc/types_nl.h"
#include "nel/misc/path.h"
#include "nel/misc/file.h"
#include "nel/misc/algo.h"
#include "nel/misc/time_nl.h"
#include "nel/misc/sheet_id.h"
#include "game_share/backup_service_messages.h"
#include "server_share/backup_service_itf.h"
#include "server_share/handy_commands.h"
#include "backup_service.h"
#include "web_connection.h"
#ifdef NL_OS_WINDOWS
# define NOMINMAX
# include <windows.h>
#endif // NL_OS_WINDOWS
// force admin module to link in
extern void admin_modules_forceLink();
void foo()
{
admin_modules_forceLink();
}
//-------------------------------------------------------------------------------------------------
// struct CBackupMsgSaveFile
// For receiving only: first construct a CBackupMsgSaveFile object, then you have access to
// the Filename and the stream
//-------------------------------------------------------------------------------------------------
struct CBackupMsgSaveFileRecv
{
// Constructor
CBackupMsgSaveFileRecv( NLMISC::IStream& streamFrom )
{
streamFrom.serial(FileName);
}
// Filename
std::string FileName;
};
extern CDirectoryRateStat DirStats;
extern NLMISC::CVariable<std::string> SaveShardRoot;
using namespace NLNET;
using namespace NLMISC;
using namespace std;
void cbReadState(IVariable&);
CVariable<std::string> MasterBSHost("backup", "MasterBSHost", "Master backup Host address", "", 0, true);
CVariable<bool> BSReadState("backup", "BSReadState", "Current read files state", false, 0, false, cbReadState);
CVariable<uint16> L3ListeningPort("backup", "L3ListeningPort", "Port used for layer 3 listen socket", 0, 0, true);
bool MasterBSUp = false;
bool BSIsSlave = false;
NLMISC::TTime LastMasterPing = 0;
bool PongReceived = false;
//-----------------------------------------------------------------------------
static void cbConnection( const string &serviceName, NLNET::TServiceId serviceId, void *arg )
{
if (serviceName == "BS" && BSIsSlave && serviceId != IService::getInstance()->getServiceId())
{
nlinfo("SLAVE BS: Master BS is up.");
MasterBSUp = true;
PongReceived = true;
IService::getInstance()->clearCurrentStatus("WaitingMaster");
IService::getInstance()->removeStatusTag("WaitingMaster");
IService::getInstance()->addStatusTag("MasterRunning");
IService::getInstance()->removeStatusTag("MasterDown");
}
// notify file manager a service connected
CBackupService::getInstance()->FileManager.notifyServiceConnection(serviceId, serviceName);
} // cbConnection //
//-----------------------------------------------------------------------------
static void cbDisconnection( const string &serviceName, NLNET::TServiceId serviceId, void *arg )
{
if (serviceName == "BS" && !MasterBSHost.get().empty())
{
nlwarning("SLAVE BS: MASTER BS IS DOWN!! File reading allowed!");
MasterBSUp = false;
BSReadState = true;
IService::getInstance()->addStatusTag("MasterDown");
IService::getInstance()->removeStatusTag("MasterRunning");
}
} // cbDisconnection //
//-----------------------------------------------------------------------------
void cbReadState(IVariable& v)
{
if (!BSIsSlave)
return;
CBackupService::getInstance()->FileManager.forbidStall(!BSReadState.get());
}
//-----------------------------------------------------------------------------
static void cbBSPing( CMessage& msgin, const std::string &serviceName, NLNET::TServiceId serviceId )
{
CMessage msgout("BS_PONG");
CUnifiedNetwork::getInstance()->send(serviceId, msgout);
}
//-----------------------------------------------------------------------------
static void cbBSPong( CMessage& msgin, const std::string &serviceName, NLNET::TServiceId serviceId )
{
// don't acknowledge my own pong
if (serviceId == IService::getInstance()->getServiceId())
return;
PongReceived = true;
}
CVariable<string> StatDirFilter("Stats", "StatDirFilter", "filter of the backup files path to be used", "save_shard", 0, true);
//-----------------------------------------------------------------------------
// cbSaveFile
//
// message format:
// - std::string: fileName
// - remaining of the stream: fileData
//
static void cbSaveFile( CMessage& msgin, const std::string &serviceName, NLNET::TServiceId serviceId )
{
try
{
CBackupMsgSaveFileRecv msg( msgin );
CWriteFile* access = new CWriteFile(msg.FileName, serviceId, 0, /*msg.Data*/msgin);
access->FailureMode = CWriteFile::MajorFailureIfFileUnwritable | CWriteFile::MajorFailureIfFileUnbackupable;
access->BackupFile = false;
access->Append = false;
CBackupService::getInstance()->FileManager.stackFileAccess(access);
}
catch (...)
{
nlwarning("WARNING: caught exception in cbSaveFile()");
}
}
//-----------------------------------------------------------------------------
// cbAppendFile
//
// message format:
// - std::string: fileName
// - remaining of the stream: fileData
//
static void cbAppendFile( CMessage& msgin, const std::string &serviceName, NLNET::TServiceId serviceId )
{
try
{
CBackupMsgSaveFileRecv msg( msgin );
CWriteFile* access = new CWriteFile(msg.FileName, serviceId, 0, /*msg.Data*/msgin);
access->FailureMode = CWriteFile::MajorFailureIfFileUnwritable | CWriteFile::MajorFailureIfFileUnbackupable;
access->BackupFile = false;
access->Append = true;
CBackupService::getInstance()->FileManager.stackFileAccess(access);
}
catch (...)
{
nlwarning("WARNING: caught exception in cbAppendFile()");
}
}
//-----------------------------------------------------------------------------
// cbAppendFileCheck
//
// message format:
// - std::string: fileName
// - remaining of the stream: fileData
//
static void cbAppendFileCheck( CMessage& msgin, const std::string &serviceName, NLNET::TServiceId serviceId )
{
try
{
CBackupMsgSaveFileRecv msg( msgin );
CWriteFile* access = new CWriteFile(msg.FileName, serviceId, 0, msgin);
access->CreateDir = true;
access->FailureMode = CWriteFile::MajorFailureIfFileUnwritable | CWriteFile::MajorFailureIfFileUnbackupable;
access->BackupFile = false;
access->Append = true;
CBackupService::getInstance()->FileManager.stackFileAccess(access);
}
catch (...)
{
nlwarning("WARNING: caught exception in cbAppendFileCheck()");
}
}
//-----------------------------------------------------------------------------
// cbLoadFile
static void cbLoadFile( CMessage& msgin, const std::string &serviceName, NLNET::TServiceId serviceId )
{
if (!BSReadState.get())
return;
try
{
CBackupMsgRequestFile msg;
msgin.serial(msg);
CLoadFile* access = new CLoadFile(msg.FileName, serviceId, msg.RequestId);
CBackupService::getInstance()->FileManager.stackFileAccess(access);
}
catch (...)
{
nlwarning("WARNING: caught exception in cbLoadFile()");
}
}
static void cbReadMode( CMessage& msgin, TSockId from, CCallbackNetBase &netbase)
{
// encode the read mode and return
CMessage msgout("BS_READ_MODE");
bool readMode = BSReadState.get();
nlWrite(msgout, serial, readMode);
// send it back to sender
netbase.send(msgout, from);
}
static void cbSyncLoadFile( CMessage& msgin, TSockId from, CCallbackNetBase &netbase)
{
if (!BSReadState.get())
return;
try
{
CBackupMsgRequestFile msg;
msgin.serial(msg);
CLoadFile* access = new CLoadFile(msg.FileName, TRequester(from, &netbase), msg.RequestId);
CBackupService::getInstance()->FileManager.stackFileAccess(access);
}
catch (...)
{
nlwarning("WARNING: caught exception in cbLoadFile()");
}
}
//-----------------------------------------------------------------------------
// cbDeleteFile
static void cbDeleteFile( CMessage& msgin, const std::string &serviceName, NLNET::TServiceId serviceId )
{
try
{
std::string fileToDelete;
msgin.serial(fileToDelete);
CDeleteFile* access = new CDeleteFile(fileToDelete, serviceId, 0);
access->BackupFile = true;
CBackupService::getInstance()->FileManager.stackFileAccess(access);
}
catch (...)
{
nlwarning("WARNING: caught exception in cbDeleteFile()");
}
}
//-----------------------------------------------------------------------------
// cbDeleteFileNoBackup
static void cbDeleteFileNoBackup( CMessage& msgin, const std::string &serviceName, NLNET::TServiceId serviceId )
{
try
{
std::string fileToDelete;
msgin.serial(fileToDelete);
CDeleteFile* access = new CDeleteFile(fileToDelete, serviceId, 0);
access->BackupFile = false;
CBackupService::getInstance()->FileManager.stackFileAccess(access);
}
catch (...)
{
nlwarning("WARNING: caught exception in cbDeleteFile()");
}
}
//-----------------------------------------------------------------------------
// cbSaveCheckFile
static void cbSaveCheckFile( CMessage& msgin, const std::string &serviceName, NLNET::TServiceId serviceId )
{
try
{
CBackupMsgSaveFileRecv msg( msgin );
CWriteFile* access = new CWriteFile(msg.FileName, serviceId, 0, /*msg.Data*/msgin);
access->FailureMode = CWriteFile::MajorFailureIfFileUnwritable | CWriteFile::MajorFailureIfFileUnbackupable;
access->BackupFile = false;
access->Append = false;
access->CreateDir = true;
CBackupService::getInstance()->FileManager.stackFileAccess(access);
}
catch (...)
{
nlwarning("WARNING: caught exception in cbSaveFile()");
}
/*
if( CBackupService::getInstance()->getStall() == true )
{
// stall shard
CBackupService::getInstance()->stallShard( std::string() );
return;
}
CBackupMsgSaveFile msg;
{
H_AUTO(SaveCheckNetSerial);
msgin.serial(msg);
}
nlinfo("SAVE: Saving file '%s' size %u", msg.FileName.c_str(), msg.Data.length());
std::string path = CFile::getPath(msg.FileName);
if ((!CFile::isExists(path) || !CFile::isDirectory(path)) && (!CFile::createDirectoryTree(path) || !CFile::setRWAccess(path)))
{
nlwarning("Can't check directory '%s' existence (to write file '%s'), shard stalled until problem are resolved !!!", path.c_str(), msg.FileName.c_str());
// stall shard
CBackupService::getInstance()->stallShard( msg.FileName );
return;
}
try
{
NLMISC::CFile::copyFile( msg.FileName + string(".backup"), msg.FileName );
}
catch(const Exception &e)
{
nlwarning("Can't write file '%s' size %u : '%s', shard stalled until problem are resolved !!!", ( msg.FileName + string(".backup") ).c_str(), msg.Data.length(), e.what() );
// stall shard
CBackupService::getInstance()->stallShard( msg.FileName );
return;
}
COFile f;
{
H_AUTO(SaveCheckFileOpen);
if(!f.open(msg.FileName))
{
nlwarning("Can't open file '%s' size %u, shard stalled until problem are resolved !!!", msg.FileName.c_str(), msg.Data.length());
// stall shard
CBackupService::getInstance()->stallShard( msg.FileName );
return;
}
}
try
{
H_AUTO(SaveCheckFileSerial);
f.serialBuffer( (uint8*)msg.Data.buffer(), msg.Data.length() );
DirStats.writeFile(msg.FileName, msg.Data.length());
}
catch(const Exception &e)
{
nlwarning("Can't write file '%s' size %u : '%s', shard stalled until problem are resolved !!!", msg.FileName.c_str(), msg.Data.length(), e.what());
// stall shard
CBackupService::getInstance()->stallShard( msg.FileName );
}
{
H_AUTO(SaveCheckFileClose);
f.close();
}
*/
}
//-----------------------------------------------------------------------------
// cbGetFileClass
struct CClassResult
{
CClassResult() : Timestamp(0) {}
CClassResult(const std::string& file, uint32 stamp) : Timestamp(stamp), File(file) {}
uint32 Timestamp;
std::string File;
bool operator < (const CClassResult& b) const { return Timestamp > b.Timestamp; }
};
static CMessage getFileClassImp( CMessage& msgin)
{
// retrieve the info from the input message
CBackupMsgFileClass inMsg;
msgin.serial(inMsg);
// setup the output message;
CBackupMsgReceiveFileClass outMsg;
outMsg.RequestId= inMsg.RequestId;
bool hasWildcard = false;
{
H_AUTO(GetFileClass_CheckWildcard);
for (uint j=0; !hasWildcard && j<inMsg.Classes.size(); ++j)
{
const CBackupFileClass& fclass = inMsg.Classes[j];
uint k;
for (k=0; k<fclass.Patterns.size(); ++k)
{
if (fclass.Patterns[k].find('*') != std::string::npos)
{
hasWildcard = true;
break;
}
}
}
}
std::vector<std::vector<CClassResult> > classes;
classes.resize(inMsg.Classes.size());
if (hasWildcard)
{
H_AUTO(GetFileClass_GetContent);
std::vector<std::string> files;
NLMISC::CPath::getPathContent(getBackupFileName(inMsg.Directory), false, false, true, files); // caution: it returns full path names
for (uint i=0; i<files.size(); ++i)
{
uint32 fstamp = CFile::getFileModificationDate(files[i]);
std::string fname = NLMISC::CFile::getFilename(files[i]);
for (uint j=0; j<inMsg.Classes.size(); ++j)
{
const CBackupFileClass& fclass = inMsg.Classes[j];
uint k;
for (k=0; k<fclass.Patterns.size(); ++k)
{
if (NLMISC::testWildCard(fname, fclass.Patterns[k]))
{
classes[j].push_back(CClassResult(files[i], fstamp));
break;
}
}
if (k < fclass.Patterns.size())
break;
}
}
}
else
{
H_AUTO(GetFileClass_GetFiles);
for (uint j=0; j<inMsg.Classes.size(); ++j)
{
const CBackupFileClass& fclass = inMsg.Classes[j];
uint k;
for (k=0; k<fclass.Patterns.size(); ++k)
{
string file = CPath::standardizePath(inMsg.Directory)+fclass.Patterns[k]; // relative filename
string rfile = getBackupFileName(file); // full filename
if (CFile::isExists(rfile))
classes[j].push_back(CClassResult(file, CFile::getFileModificationDate(rfile)));
}
}
}
CFileDescriptionContainer& fdc = outMsg.Fdc;
for (uint i=0; i<classes.size(); ++i)
{
std::sort(classes[i].begin(), classes[i].end());
for (uint j=0; j<classes[i].size(); ++j)
{
std::string rfile = classes[i][j].File;
// if there's wildcard, the file is already full so we don't need to add again the backup path or we ll have "save_shard/save_shard/..." that is not valid
if (!CFile::isExists(rfile))
rfile = getBackupFileName(classes[i][j].File);
fdc.addFile(classes[i][j].File, CFile::getFileModificationDate(rfile),CFile::getFileSize(rfile));
}
}
// In case something like getPathContent() has returned full paths, make paths relative to match the requested filenames
fdc.stripFilename(SaveShardRoot.get());
// compose the output message
CMessage msgout("BS_FILE_CLASS");
msgout.serial(outMsg);
return msgout;
}
static void cbGetFileClass( CMessage& msgin, const std::string &serviceName, NLNET::TServiceId serviceId )
{
if (!BSReadState.get())
return;
CMessage msgOut = getFileClassImp(msgin);
// send the output message
CUnifiedNetwork::getInstance()->send(serviceId, msgOut);
}
static void cbSyncGetFileClass( CMessage& msgin, TSockId from, CCallbackNetBase &netbase)
{
if (!BSReadState.get())
return;
CMessage msgOut = getFileClassImp(msgin);
// send the output message
netbase.send(msgOut, from);
}
//-----------------------------------------------------------------------------
// cbAppend
static void cbAppend( CMessage& msgin, const std::string &serviceName, NLNET::TServiceId serviceId )
{
try
{
CBackupMsgAppend inMsg;
msgin.serial(inMsg);
std::string append = inMsg.Append+'\n';
uint8* data = (uint8*)(const_cast<char*>(append.c_str()));
uint dataSize = (uint)append.size();
CWriteFile* access = new CWriteFile(inMsg.FileName, serviceId, 0, data, dataSize);
access->FailureMode = CWriteFile::MajorFailureIfFileUnwritable;
access->BackupFile = false;
access->Append = true;
CBackupService::getInstance()->FileManager.stackFileAccess(access);
}
catch (...)
{
nlwarning("WARNING: caught exception in cbAppendFile()");
}
}
//-----------------------------------------------------------------------------
bool CBackupService::update()
{
TTime ptime = CTime::getLocalTime();
if (BSIsSlave && MasterBSUp && ptime - LastMasterPing > 60*1000)
{
CMessage msgout("BS_PING");
CUnifiedNetwork::getInstance()->send("BS", msgout);
PongReceived = false;
LastMasterPing = ptime;
}
FileManager.update();
updateWebConnection();
_CallbackServer->update();
if (BSReadState == true)
{
IService::getInstance()->addStatusTag("ReadWrite");
IService::getInstance()->removeStatusTag("WriteOnly");
}
else
{
IService::getInstance()->removeStatusTag("ReadWrite");
IService::getInstance()->addStatusTag("WriteOnly");
}
return true;
}
//-----------------------------------------------------------------------------
void CBackupService::release()
{
FileManager.release();
releaseWebConnection();
delete _CallbackServer;
}
//-----------------------------------------------------------------------------
void CBackupService::stallShard( const std::string& fileName )
{
/* CMessage msgOut("STALL_SHARD");
std::string s = fileName;
msgOut.serial( s );
CUnifiedNetwork::getInstance()->send("EGS", msgOut );
_SaveStall = true;
*/
fileName.empty() ? nlwarning("BackupService are in stall mode !") : nlwarning("Backup service enter in stall mode when trying save file %s", fileName.c_str() );
}
void CBackupService::onModuleDown(NLNET::IModuleProxy *proxy)
{
}
//-----------------------------------------------------------------------------
TUnifiedCallbackItem CbArray[]=
{
{ "save_file", cbSaveFile },
{ "load_file", cbLoadFile },
{ "append_file", cbAppendFile },
{ "append_file_check", cbAppendFileCheck },
{ "SAVE_CHECK_FILE", cbSaveCheckFile },
{ "DELETE_FILE", cbDeleteFile },
{ "DELETE_FILE_NO_BACKUP", cbDeleteFileNoBackup },
{ "GET_FILE_CLASS", cbGetFileClass },
{ "APPEND", cbAppend },
{ "BS_PING", cbBSPing },
{ "BS_PONG", cbBSPong },
};
TCallbackItem cbSyncArray[] =
{
{ "GET_READ_MODE", cbReadMode }, // TODO : implement me !
{ "load_file", cbSyncLoadFile },
{ "GET_FILE_CLASS", cbSyncGetFileClass },
};
//-----------------------------------------------------------------------------
void CBackupService::init()
{
FileManager.init();
setUpdateTimeout(100);
_SaveStall = false;
// set the connection and disconnection callbacks
CUnifiedNetwork::getInstance()->setServiceUpCallback( string("*"), cbConnection, 0);
CUnifiedNetwork::getInstance()->setServiceDownCallback( string("*"), cbDisconnection, 0);
CUnifiedNetwork::getInstance()->setServiceUpCallback( string("BS"), cbConnection, 0);
CUnifiedNetwork::getInstance()->setServiceDownCallback( string("BS"), cbDisconnection, 0);
// Init the sheet Id
CSheetId::init(false);
if (!MasterBSHost.get().empty())
{
IService::getInstance()->addStatusTag("SlaveMode");
IService::getInstance()->setCurrentStatus("WaitingMaster");
BSIsSlave = true;
FileManager.forbidStall();
// I'm a slave, try to contact master
string host = MasterBSHost;
if (host.find (":") == string::npos)
host += ":49990";
CUnifiedNetwork::getInstance()->addService ("BS", CInetAddress(host));
}
// set the initial read state from the config file
CConfigFile::CVar *readState = ConfigFile.getVarPtr("BSReadState");
if (readState != NULL)
BSReadState = readState->asBool();
initWebConnection();
_CallbackServer = new NLNET::CCallbackServer;
_CallbackServer->addCallbackArray(cbSyncArray, sizeofarray(cbSyncArray));
// open the layer 3 callback server if required
if (L3ListeningPort != 0)
_CallbackServer->init(L3ListeningPort);
}
static const char* getCompleteServiceName(const IService* theService, const char *defaultName)
{
static std::string s;
s= defaultName;
if (theService->haveLongArg("name"))
{
s+= "_"+theService->getLongArg("name");
}
if (theService->haveLongArg("fullname"))
{
s= theService->getLongArg("fullname");
}
return s.c_str();
}
static const char* getShortServiceName(const IService* theService, const char *defaultName)
{
static std::string s;
s= defaultName;
if (theService->haveLongArg("shortname"))
{
s= theService->getLongArg("shortname");
}
return s.c_str();
}
NLNET_SERVICE_MAIN( CBackupService, getShortServiceName(scn, "BS"), getCompleteServiceName(scn, "backup_service"), 49990, CbArray, "", "" )
void CDirectoryRateStat::clear()
{
TDirectoryMap::iterator first = _DirectoryMap.begin(), last = _DirectoryMap.end();
for (; first != last; ++first)
(*first).second.clear();
}
void CDirectoryRateStat::readFile(const std::string& filename, uint32 filesize)
{
NLMISC::TTime now = NLMISC::CTime::getLocalTime();
if (filename.find("www") != std::string::npos)
{
_DirectoryMap["www"].read(now, filesize);
}
else if (filename.find(StatDirFilter.get()) != std::string::npos)
{
_DirectoryMap[NLMISC::CFile::getPath(filename)].read(now, filesize);
}
}
void CDirectoryRateStat::writeFile(const std::string& filename, uint32 filesize)
{
NLMISC::TTime now = NLMISC::CTime::getLocalTime();
if (filename.find("www") != std::string::npos)
{
_DirectoryMap["www"].write(now, filesize);
}
else if (filename.find(StatDirFilter.get()) != std::string::npos)
{
_DirectoryMap[NLMISC::CFile::getPath(filename)].write(now, filesize);
}
}
uint CDirectoryRateStat::getMeanReadRate()
{
NLMISC::TTime limit = NLMISC::CTime::getLocalTime()-60*1000;
uint64 read = 0;
TDirectoryMap::iterator first = _DirectoryMap.begin(), last = _DirectoryMap.end();
for (; first != last; ++first)
{
(*first).second.updateTime(limit);
read += (*first).second.ReadBytes;
}
return (uint)(read / 60);
}
uint CDirectoryRateStat::getMeanWriteRate()
{
NLMISC::TTime limit = NLMISC::CTime::getLocalTime()-60*1000;
uint64 write = 0;
TDirectoryMap::iterator first = _DirectoryMap.begin(), last = _DirectoryMap.end();
for (; first != last; ++first)
{
(*first).second.updateTime(limit);
write += (*first).second.WrittenBytes;
}
return (uint)(write / 60);
}
void CDirectoryRateStat::display(NLMISC::CLog& log)
{
uint pathsize = 0;
TDirectoryMap::iterator first = _DirectoryMap.begin(), last = _DirectoryMap.end();
for (; first != last; ++first)
if ((*first).first.size() > pathsize)
pathsize = (uint)(*first).first.size();
NLMISC::TTime limit = NLMISC::CTime::getLocalTime()-60*1000;
std::string format = "%-"+NLMISC::toString(pathsize)+"s %6s %10s %6s %10s";
log.displayNL(format.c_str(), "directory", "rdfile", "read", "wrfile", "write");
for (first=_DirectoryMap.begin(); first != last; ++first)
{
(*first).second.updateTime(limit);
uint64 rdrate = (*first).second.ReadBytes/60;
uint64 wrrate = (*first).second.WrittenBytes/60;
log.displayNL(format.c_str(),
(*first).first.c_str(),
NLMISC::toString((*first).second.ReadFiles).c_str(),
(NLMISC::bytesToHumanReadable(uint32(rdrate))+"/s").c_str(),
NLMISC::toString((*first).second.WrittenFiles).c_str(),
(NLMISC::bytesToHumanReadable(uint32(wrrate))+"/s").c_str());
}
}