2012-05-29 13:31:11 +00:00
// 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/common.h>
# include <nel/misc/variable.h>
# include <nel/misc/command.h>
# include <nel/misc/algo.h>
# include <nel/misc/path.h>
# include <nel/net/service.h>
//#include <nel/net/buf_server.h>
using namespace std ;
using namespace NLMISC ;
using namespace NLNET ;
/// Callback type for web connection
typedef void ( * WebCallback ) ( CMemStream & msgin , TSockId host ) ;
CBufServer * WebServer = NULL ;
void initWebConnection ( )
{
nlassert ( WebServer = = NULL ) ;
uint16 port = ( uint16 ) IService : : getInstance ( ) - > ConfigFile . getVar ( " WebPort " ) . asInt ( ) ;
// if the WebPort is set to 0 then we don't initialise the web server
if ( port = = 0 )
return ;
nlinfo ( " Initialise Web socket on port %d " , port ) ;
WebServer = new CBufServer ( ) ;
nlassert ( WebServer ! = NULL ) ;
WebServer - > init ( port ) ;
}
void cbExecute ( CMemStream & msgin , TSockId host )
{
std : : string str ;
msgin . serial ( str ) ;
ICommand : : execute ( str , * NLMISC : : InfoLog ) ;
}
void cbGetSaveList ( CMemStream & msgin , TSockId host ) ;
void cbRestoreSave ( CMemStream & msgin , TSockId host ) ;
void cbCopyOverSave ( CMemStream & msgin , TSockId host ) ;
WebCallback WebCallbackArray [ ] =
{
cbExecute ,
cbGetSaveList ,
cbRestoreSave ,
cbCopyOverSave ,
} ;
void updateWebConnection ( )
{
nlassert ( WebServer ! = NULL ) ;
try
{
WebServer - > update ( ) ;
while ( WebServer - > dataAvailable ( ) )
{
// create a string mem stream to easily communicate with web server
NLMISC : : CMemStream msgin ( true ) ;
TSockId host ;
bool success = false ;
std : : string reason ;
try
{
WebServer - > receive ( msgin , & host ) ;
uint32 fake = 0 ;
msgin . serial ( fake ) ;
uint8 messageType = 0xff ;
msgin . serial ( messageType ) ;
if ( messageType < sizeof ( WebCallbackArray ) / sizeof ( WebCallbackArray [ 0 ] ) )
{
WebCallbackArray [ messageType ] ( msgin , host ) ;
success = true ;
}
}
catch ( const Exception & e )
{
nlwarning ( " Error during receiving: '%s' " , e . what ( ) ) ;
reason = e . what ( ) ;
}
if ( ! success )
{
nlwarning ( " Failed to decode Web command " ) ;
CMemStream msgout ;
uint32 fake = 0 ;
msgout . serial ( fake ) ;
std : : string result = " 0:Failed to decode command " ;
if ( ! reason . empty ( ) )
result + = " ( " + reason + " ) " ;
msgout . serial ( result ) ;
WebServer - > send ( msgout , host ) ;
}
}
}
catch ( const Exception & e )
{
nlwarning ( " Error during update: '%s' " , e . what ( ) ) ;
}
}
void releaseWebConnection ( )
{
nlassert ( WebServer ! = NULL ) ;
delete WebServer ;
WebServer = 0 ;
}
/*
*
* Callbacks
*
*/
// Automatic SaveShardRoot path standardization
void cbOnSaveShardRootModified ( NLMISC : : IVariable & var )
{
var . fromString ( CPath : : standardizePath ( var . toString ( ) ) ) ;
}
CVariable < string > IncrementalBackupDirectory ( " backup " , " IncrementalBackupDirectory " , " Directory to find incremental backuped archives " , " " , 0 , true ) ;
// (SaveShardRoot from game_share/backup_service_interface.cpp is not instanciated because the nothing is used from that file)
extern CVariable < string > SaveShardRoot ;
CVariable < string > SaveTemplatePath ( " backup " , " SaveTemplatePath " , " Directory to find saves (with shard and account replacement strings) " , " $shard/characters/account_$userid_$charid$ext " , 0 , true ) ;
CVariable < string > SaveExtList ( " backup " , " SaveExtList " , " List of possible extensions for save files (space separated) " , " _pdr.bin _pdr.xml .bin " , 0 , true ) ;
string checkFile ( const string & templatePath , const string & shard , const string & userid , const string & charid , const vector < string > & extensions )
{
string result ;
uint i ;
for ( i = 0 ; i < extensions . size ( ) ; + + i )
{
string file = templatePath ;
strFindReplace ( file , string ( " $shard " ) , shard ) ;
strFindReplace ( file , string ( " $userid " ) , userid ) ;
strFindReplace ( file , string ( " $charid " ) , charid ) ;
strFindReplace ( file , string ( " $ext " ) , extensions [ i ] ) ;
if ( CFile : : fileExists ( file ) )
{
result + = file + " : " +
toString ( CFile : : getFileModificationDate ( file ) ) + " : " +
toString ( CFile : : getFileSize ( file ) ) + " \n " ;
}
}
return result ;
}
void cbGetSaveList ( CMemStream & msgin , TSockId host )
{
string str ;
msgin . serial ( str ) ;
vector < string > params ;
explode ( str , string ( " %% " ) , params , true ) ;
2015-02-23 11:37:51 +00:00
string incrementalDir = IncrementalBackupDirectory . get ( ) ;
string saveShardRoot = SaveShardRoot . get ( ) ;
string templatePath = SaveTemplatePath . get ( ) ;
string extList = SaveExtList . get ( ) ;
2012-05-29 13:31:11 +00:00
string shard ;
string userid ;
string charid ;
uint i ;
for ( i = 0 ; i < params . size ( ) ; + + i )
{
vector < string > param ;
explode ( params [ i ] , string ( " = " ) , param , false ) ;
if ( param . empty ( ) )
continue ;
string var = param [ 0 ] ;
string val ;
if ( param . size ( ) > 1 )
val = param [ 1 ] ;
if ( var = = " incrementalDir " )
incrementalDir = val ;
if ( var = = " saveShardRoot " )
saveShardRoot = val ;
if ( var = = " templatePath " )
templatePath = val ;
if ( var = = " extList " )
extList = val ;
if ( var = = " shard " )
shard = val ;
if ( var = = " userid " )
userid = val ;
if ( var = = " charid " )
charid = val ;
}
incrementalDir = CPath : : standardizePath ( incrementalDir ) ;
saveShardRoot = CPath : : standardizePath ( saveShardRoot ) ;
vector < string > extensions ;
explode ( extList , string ( " " ) , extensions , false ) ;
string result = checkFile ( saveShardRoot + templatePath , shard , userid , charid , extensions ) ;
vector < string > incrementalDirectories ;
CPath : : getPathContent ( incrementalDir , false , true , false , incrementalDirectories ) ;
if ( ! incrementalDirectories . empty ( ) )
{
std : : sort ( incrementalDirectories . begin ( ) , incrementalDirectories . end ( ) ) ;
for ( i = ( uint ) incrementalDirectories . size ( ) - 1 ; ( sint ) i > = 0 ; - - i )
{
string p = CPath : : standardizePath ( incrementalDirectories [ i ] , true ) ;
// avoid double / inside path
p + = ( templatePath [ 0 ] = = ' / ' ? templatePath . substr ( 1 ) : templatePath ) ;
result + = checkFile ( p , shard , userid , charid , extensions ) ;
}
}
CMemStream msgout ;
uint32 fake = 0 ;
msgout . serial ( fake ) ;
result = " 1: " + result ;
msgout . serial ( result ) ;
WebServer - > send ( msgout , host ) ;
}
void cbRestoreSave ( CMemStream & msgin , TSockId host )
{
string str ;
msgin . serial ( str ) ;
vector < string > params ;
explode ( str , string ( " %% " ) , params , true ) ;
2015-02-23 11:37:51 +00:00
string saveShardRoot = SaveShardRoot . get ( ) ;
string templatePath = SaveTemplatePath . get ( ) ;
2012-05-29 13:31:11 +00:00
string shard ;
string userid ;
string charid ;
string file ;
uint i ;
for ( i = 0 ; i < params . size ( ) ; + + i )
{
vector < string > param ;
explode ( params [ i ] , string ( " = " ) , param , false ) ;
if ( param . empty ( ) )
continue ;
string var = param [ 0 ] ;
string val ;
if ( param . size ( ) > 1 )
val = param [ 1 ] ;
if ( var = = " saveShardRoot " )
saveShardRoot = val ;
if ( var = = " templatePath " )
templatePath = val ;
if ( var = = " shard " )
shard = val ;
if ( var = = " userid " )
userid = val ;
if ( var = = " charid " )
charid = val ;
if ( var = = " file " )
file = val ;
}
saveShardRoot = CPath : : standardizePath ( saveShardRoot ) ;
string outputfile = CPath : : standardizePath ( CFile : : getPath ( saveShardRoot + templatePath ) ) + CFile : : getFilename ( file ) ;
strFindReplace ( outputfile , string ( " $shard " ) , shard ) ;
strFindReplace ( outputfile , string ( " $userid " ) , userid ) ; // just in case...
strFindReplace ( outputfile , string ( " $charid " ) , charid ) ; //
bool success ;
success = CFile : : copyFile ( outputfile , file , false ) ;
CMemStream msgout ;
uint32 fake = 0 ;
msgout . serial ( fake ) ;
string result ;
if ( success )
{
result = " 1:Restore " + outputfile + " succeded. " ;
}
else
{
result = " 0:Failed to restore " + outputfile + " , copy failed. " ;
}
msgout . serial ( result ) ;
WebServer - > send ( msgout , host ) ;
}
void cbCopyOverSave ( CMemStream & msgin , TSockId host )
{
string str ;
msgin . serial ( str ) ;
vector < string > params ;
explode ( str , string ( " %% " ) , params , true ) ;
2015-02-23 11:37:51 +00:00
string saveShardRoot = SaveShardRoot . get ( ) ;
string templatePath = SaveTemplatePath . get ( ) ;
string extList = SaveExtList . get ( ) ;
2012-05-29 13:31:11 +00:00
string shard ;
string userid ;
string charid ;
string file ;
uint i ;
for ( i = 0 ; i < params . size ( ) ; + + i )
{
vector < string > param ;
explode ( params [ i ] , string ( " = " ) , param , false ) ;
if ( param . empty ( ) )
continue ;
string var = param [ 0 ] ;
string val ;
if ( param . size ( ) > 1 )
val = param [ 1 ] ;
if ( var = = " saveShardRoot " )
saveShardRoot = val ;
if ( var = = " templatePath " )
templatePath = val ;
if ( var = = " extList " )
extList = val ;
if ( var = = " shard " )
shard = val ;
if ( var = = " userid " )
userid = val ;
if ( var = = " charid " )
charid = val ;
if ( var = = " file " )
file = val ;
}
saveShardRoot = CPath : : standardizePath ( saveShardRoot ) ;
vector < string > extensions ;
explode ( extList , string ( " " ) , extensions , false ) ;
for ( i = 0 ; i < extensions . size ( ) ; + + i )
{
if ( file . size ( ) > = extensions [ i ] . size ( ) & & file . substr ( file . size ( ) - extensions [ i ] . size ( ) ) = = extensions [ i ] )
{
break ;
}
}
bool success = false ;
string result ;
if ( i < extensions . size ( ) )
{
string outputfile = CPath : : standardizePath ( saveShardRoot ) + templatePath ;
strFindReplace ( outputfile , string ( " $shard " ) , shard ) ;
strFindReplace ( outputfile , string ( " $userid " ) , userid ) ;
strFindReplace ( outputfile , string ( " $charid " ) , charid ) ;
strFindReplace ( outputfile , string ( " $ext " ) , extensions [ i ] ) ;
success = CFile : : copyFile ( outputfile , file , false ) ;
if ( ! success )
result = " Failed to copy " + file + " over " + outputfile + " . " ;
}
else
{
result = " failed to match valid extension " ;
}
result = string ( success ? " 1: " : " 0: " ) + result ;
CMemStream msgout ;
uint32 fake = 0 ;
msgout . serial ( fake ) ;
msgout . serial ( result ) ;
WebServer - > send ( msgout , host ) ;
}