2010-05-06 00:08:41 +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 "stdpch.h"
# include "nel/net/service.h"
# include "nel/net/message.h"
# include "nel/misc/file.h"
# include "nel/misc/path.h"
# include "nel/misc/o_xml.h"
# include "nel/misc/i_xml.h"
# include "nel/misc/algo.h"
# include "nel/misc/sstring.h"
# include "nel/misc/random.h"
# include "game_share/backup_service_interface.h"
# include "game_share/utils.h"
# include "name_manager.h"
# include "database_mapping.h"
using namespace std ;
using namespace NLMISC ;
using namespace NLNET ;
using namespace MSW ;
using namespace CHARSYNC ;
CVariable < bool > NameManagerCheckOnStatup ( " name_mgr " , " NameManagerCheckOnStatup " , " If true name manager will do a complete check on character name and rename any invalid one " , true , 0 , true ) ;
NLMISC : : CRandom RandomGenerator ;
const char * GUILD_NAME_FILE = " guild_names.txt " ;
//-----------------------------------------------------------------------------
bool CNameManager : : assignName ( uint32 charId , const ucstring & ucName , uint32 homeSessionId , bool skipTest )
{
// if (eId == NLMISC::CEntityId::Unknown)
// return false;
const TCharSlot charSlot ( charId ) ;
const uint32 playerId = charSlot . UserId ;
const uint8 charIndex = charSlot . CharIndex ;
// check that name is usable
if ( ! skipTest & & isNameUsable ( ucName , playerId , charIndex , homeSessionId ) ! = TCharacterNameResult : : cnr_ok )
return false ;
// assign name and save
const string name = toLower ( ucName . toUtf8 ( ) ) ;
// remove any temporary reserve for this name
_TemporaryReservedNames . erase ( name ) ;
TFullName fullname ( name , homeSessionId ) ;
const TCharSlot * owner = _Names . getB ( fullname ) ;
if ( owner ! = NULL & & * owner = = charSlot )
// association already exist, nothing more to do
return true ;
else if ( owner ! = NULL )
{
// we need to remove this assoc
_Names . removeWithA ( fullname ) ;
_ReleasedNames . insert ( ( owner - > UserId < < 4 ) + owner - > CharIndex ) ;
}
if ( _Names . getBToAMap ( ) . find ( charSlot ) ! = _Names . getBToAMap ( ) . end ( ) )
{
// the character is associated to an other name
_Names . removeWithB ( charSlot ) ;
}
// insert the new association
_Names . add ( fullname , charSlot ) ;
_ChangedNames . insert ( charId ) ;
// saveCharacterNames();
nlinfo ( " NAMEMGR: assigned name '%s' to char %u " , name . c_str ( ) , charId ) ;
return true ;
}
//-----------------------------------------------------------------------------
void CNameManager : : liberateName ( uint32 charId , const ucstring & ucName )
{
const TCharSlot charSlot ( charId ) ;
const string name = toLower ( ucName . toUtf8 ( ) ) ;
// remove the name from the reservation if it belong to the specified user
{
TTempReservedNames : : iterator it ( _TemporaryReservedNames . find ( name ) ) ;
if ( it ! = _TemporaryReservedNames . end ( ) & & it - > second . UserId = = ( charId > > 4 ) )
{
_TemporaryReservedNames . erase ( it ) ;
}
}
const TFullName * fullname = _Names . getA ( charId ) ;
if ( fullname = = NULL | | fullname - > Name ! = ucName . toUtf8 ( ) )
{
nlwarning ( " NAMEMGR: char %u is trying to liberate a name he does not own: '%s' " , charId , name . c_str ( ) ) ;
return ;
}
_Names . removeWithB ( charSlot ) ;
_ReleasedNames . insert ( charId ) ;
// saveCharacterNames();
nlinfo ( " NAMEMGR: char %u liberated his name '%s' " , charId , name . c_str ( ) ) ;
}
//-----------------------------------------------------------------------------
// liberate any name associated to a character
void CNameManager : : liberateName ( uint32 charId )
{
const TCharSlot charSlot ( charId ) ;
TNamesIndex : : TBToAMap : : const_iterator it = _Names . getBToAMap ( ) . find ( charSlot ) ;
if ( it = = _Names . getBToAMap ( ) . end ( ) )
return ;
nlinfo ( " NAMEMGR: char %u liberated his name '%s' " , charId , it - > second . Name . c_str ( ) ) ;
// ok, association found, remove it
_Names . removeWithB ( charSlot ) ;
_ReleasedNames . insert ( charId ) ;
// saveCharacterNames();
}
//-----------------------------------------------------------------------------
//void CNameManager::checkCharacterSlot(uint32 charId, const ucstring & ucName)
//{
//// if (eId == CEntityId::Unknown)
//// return;
//
// const TCharSlot charSlot( charId );
// const TName name = toLower( ucName.toUtf8() );
//
// // verify that name is assigned to the character
// {
// TNamesIndex::TAToBMap::const_iterator it = _Names.getAToBMap().find( name );
// if ( it != _Names.getAToBMap().end() )
// {
// if( it->second != charSlot )
// {
// nlwarning("NAMEMGR: char %u has name '%s', but this name is assigned to another character", charId, name.c_str() );
// renameCharacter(charId);
// }
// }
// else
// {
// nlwarning("NAMEMGR: char %u has name '%s', but this name is not assigned", charId, name.c_str() );
// assignName(charId, ucName);
// }
// }
//
// // if this slot is not duplicated, just return
// TDuplicatedSlots::iterator it = _DuplicatedSlots.find( charSlot );
// if ( it == _DuplicatedSlots.end() )
// return;
//
// const std::set<TName> & dupNames = (*it).second;
//
// // erase other names associated to the same slot
// std::set<TName>::const_iterator itDupName;
// for (itDupName = dupNames.begin(); itDupName != dupNames.end(); ++itDupName)
// {
// const TName & dupName = (*itDupName);
// if (dupName != name)
// {
// const TCharSlot *cs = _Names.getB(dupName);
// if (cs != NULL)
// {
// _ReleasedNames.insert((cs->UserId<<4)+cs->CharIndex);
// }
// _Names.removeWithA( dupName );
// }
// }
//
// // save changes
// saveCharacterNames();
//
// // slot has been fixed
// _DuplicatedSlots.erase( it );
//
// nlinfo("NAMEMGR: fixed character slot: userId=%u, slot=%u, name='%s'", charSlot.UserId, charSlot.CharIndex, name.c_str() );
//
// if ( !_DuplicatedSlots.empty() )
// nlinfo("NAMEMGR: there still are %u duplicated character slots", _DuplicatedSlots.size() );
//}
//-----------------------------------------------------------------------------
ucstring CNameManager : : renameCharacter ( uint32 charId , uint32 homeSessionId )
{
const ucstring ucDefaultName = generateDefaultName ( charId , homeSessionId ) ;
nlverify ( assignName ( charId , ucDefaultName , homeSessionId , true ) ) ;
return ucDefaultName ;
}
//-----------------------------------------------------------------------------
ucstring CNameManager : : generateDefaultName ( uint32 charId , uint32 homeSessionId )
{
const TCharSlot charSlot ( charId ) ;
string defaultName ;
string randomLetters ( 5 , ' a ' ) ;
uint count = 0 ;
while ( 1 )
{
for ( uint i = 0 ; i < randomLetters . size ( ) ; i + + )
{
randomLetters [ i ] = ' a ' + ( char ) RandomGenerator . rand ( uint16 ( ' z ' - ' a ' ) ) ;
}
defaultName = " Default " + randomLetters ;
if ( isNameUsable ( defaultName , charSlot . UserId , charSlot . CharIndex , homeSessionId ) = = TCharacterNameResult : : cnr_ok )
break ;
// anti freeze
if ( + + count > 100 )
break ;
}
return defaultName ;
}
//-----------------------------------------------------------------------------
ucstring CNameManager : : generateDefaultGuildName ( uint32 guildId )
{
string defaultName ;
string randomLetters ( 5 , ' a ' ) ;
uint count = 0 ;
while ( 1 )
{
for ( uint i = 0 ; i < randomLetters . size ( ) ; i + + )
{
randomLetters [ i ] = ' a ' + ( char ) RandomGenerator . rand ( uint16 ( ' z ' - ' a ' ) ) ;
}
defaultName = " DefaultGuild " + randomLetters ;
if ( isGuildNameUsable ( defaultName , guildId ) = = TCharacterNameResult : : cnr_ok )
break ;
// anti freeze
if ( + + count > 100 )
break ;
}
return defaultName ;
}
//-----------------------------------------------------------------------------
/** This methods implemented by CCommandHandler is used by the
* command registry to retrieve the name of the object instance .
*/
const std : : string & CNameManager : : getCommandHandlerName ( ) const
{
const static string name = " nameManager " ;
return name ;
}
//-----------------------------------------------------------------------------
CHARSYNC : : TCharacterNameResult CNameManager : : isNameUsable ( const ucstring & ucNameIn , uint32 userId , uint8 charIndex , uint32 homeSessionId )
{
// WARNING: if you change validity checks here,
// please also change them in CEntityIdTranslator::isValidEntityName() (nel/misc/eid_translator.cpp)
// if the name contains a shard specification, remove it first
ucstring ucName ;
ucstring : : size_type pos = ucNameIn . find ( ' ( ' ) ;
if ( pos ! = ucstring : : npos )
{
//only keep the simple name for the test
ucName = ucNameIn . substr ( 0 , pos ) ;
}
else
{
// test the whole input name
ucName = ucNameIn ;
}
// perform a first validity check on the name length
if ( ucName . size ( ) < 3 )
{
nldebug ( " VALID_NAME::CNameManager::isNameUsable name '%s' rejected because is too short " , ucName . toString ( ) . c_str ( ) ) ;
return TCharacterNameResult : : cnr_invalid_name ;
}
// perform a first validity check on the name length
if ( ucName . size ( ) > 15 )
{
nldebug ( " VALID_NAME::CNameManager::isNameUsable name '%s' rejected because is too long " , ucName . toString ( ) . c_str ( ) ) ;
return TCharacterNameResult : : cnr_invalid_name ;
}
// make sure the name is only composed of valid characters
for ( uint32 i = 0 ; i < ucName . size ( ) ; + + i )
{
if ( ( ucName [ i ] < ( uint16 ) ' A ' | | ucName [ i ] > ( uint16 ) ' Z ' ) & & ( ucName [ i ] < ( uint16 ) ' a ' | | ucName [ i ] > ( uint16 ) ' z ' ) )
{
nldebug ( " VALID_NAME::CNameManager::isNameUsable name '%s' rejected because it contains invalid character " , ucName . toString ( ) . c_str ( ) ) ;
return TCharacterNameResult : : cnr_invalid_name ;
}
}
// it's now safe to convert the name to 8 bit lower case
const string name = toLower ( ucName . toUtf8 ( ) ) ;
// make sure the name isn't forbidden
for ( uint32 i = 0 ; i < _ForbiddenNames . size ( ) ; + + i )
{
if ( NLMISC : : testWildCard ( name , _ForbiddenNames [ i ] ) )
{
nldebug ( " VALID_NAME::CNameManager::isNameUsable name '%s' rejected because it contains the forbidden string '%s' " ,
ucName . toString ( ) . c_str ( ) ,
_ForbiddenNames [ i ] . c_str ( ) ) ;
return TCharacterNameResult : : cnr_invalid_name ;
}
}
// make sure the name isn't reserved
TReservedNames : : iterator rit = _ReservedNames . find ( name ) ;
if ( rit ! = _ReservedNames . end ( ) )
{
if ( ( * rit ) . second ! = userId )
{
nldebug ( " VALID_NAME::CNameManager::isNameUsable name '%s' rejected because it's reserved " , ucName . toString ( ) . c_str ( ) ) ;
return TCharacterNameResult : : cnr_already_exist ;
}
}
// make sure the name isn't temporary reserved
{
TTempReservedNames : : iterator rit = _TemporaryReservedNames . find ( name ) ;
if ( rit ! = _TemporaryReservedNames . end ( ) )
{
if ( rit - > second . UserId ! = userId & & rit - > second . UserId = = homeSessionId )
{
nldebug ( " VALID_NAME::CNameManager::isNameUsable name '%s' rejected because it's temporary reserved " , ucName . toString ( ) . c_str ( ) ) ;
return TCharacterNameResult : : cnr_already_exist ;
}
}
}
TFullName fullname ( name , homeSessionId ) ;
// make sure the name isn't used by another character
TNamesIndex : : TAToBMap : : const_iterator it = _Names . getAToBMap ( ) . find ( fullname ) ;
if ( it ! = _Names . getAToBMap ( ) . end ( ) )
{
if ( it - > second . UserId ! = userId | | it - > second . CharIndex ! = charIndex )
{
nldebug ( " VALID_NAME::CNameManager::isNameUsable name '%s' rejected because it's already used " , ucName . toString ( ) . c_str ( ) ) ;
return TCharacterNameResult : : cnr_already_exist ;
}
}
// make sure the name is not used by a guild
{
TGuildNames : : iterator it = _GuildNames . find ( name ) ;
if ( it ! = _GuildNames . end ( ) )
{
nldebug ( " VALID_NAME::CNameManager::isNameUsable name '%s' rejected because it's a guild name " , ucName . toString ( ) . c_str ( ) ) ;
return TCharacterNameResult : : cnr_already_exist ;
}
}
// the name is valid, insert it in the temporary reserve
TTemporaryReservedNameInfo trni ;
trni . UserId = userId ;
trni . ReserveDate = CTime : : getSecondsSince1970 ( ) ;
trni . HomeSessionId = homeSessionId ;
_TemporaryReservedNames . insert ( make_pair ( name , trni ) ) ;
nldebug ( " VALID_NAME::CNameManager::isNameUsable name '%s' accepted " , ucName . toString ( ) . c_str ( ) ) ;
return TCharacterNameResult : : cnr_ok ;
}
//-----------------------------------------------------------------------------
TCharacterNameResult CNameManager : : isGuildNameUsable ( const ucstring & ucName , uint32 guildId )
{
// perform a first validity check on the name length
if ( ucName . size ( ) < 3 )
return TCharacterNameResult : : cnr_invalid_name ;
// perform a first validity check on the name length
if ( ucName . size ( ) > 50 )
return TCharacterNameResult : : cnr_invalid_name ;
// make sure the name is only composed of valid characters
bool prevBlank = false ;
for ( uint i = 0 ; i < ucName . size ( ) ; i + + )
{
if ( ucName [ i ] = = ucchar ( ' ' ) )
{
if ( prevBlank )
{
return TCharacterNameResult : : cnr_invalid_name ;
}
prevBlank = true ;
}
else
{
prevBlank = false ;
if ( ! isalpha ( ucName [ i ] ) )
{
return TCharacterNameResult : : cnr_invalid_name ;
}
}
}
// it's now safe to convert the name to 8 bit lower case
const string name = toLower ( ucName . toUtf8 ( ) ) ;
// make sure the name isn't forbidden
for ( uint32 i = 0 ; i < _ForbiddenNames . size ( ) ; + + i )
{
if ( NLMISC : : testWildCard ( name , _ForbiddenNames [ i ] ) )
return TCharacterNameResult : : cnr_invalid_name ;
}
// make sure the name isn't reserved
TReservedNames : : iterator rit = _ReservedNames . find ( name ) ;
if ( rit ! = _ReservedNames . end ( ) )
{
return TCharacterNameResult : : cnr_already_exist ;
}
// make sure the name isn't temporary reserved
{
TTempReservedNames : : iterator rit = _TemporaryReservedNames . find ( name ) ;
if ( rit ! = _TemporaryReservedNames . end ( ) )
{
return TCharacterNameResult : : cnr_already_exist ;
}
}
// make sure the name isn't used by a character
TNamesIndex : : TAToBMap : : const_iterator it = _Names . getAToBMap ( ) . lower_bound ( TFullName ( name , 0 ) ) ;
if ( it ! = _Names . getAToBMap ( ) . end ( ) & & it - > first . Name = = name )
{
return TCharacterNameResult : : cnr_already_exist ;
}
// make sure the name is not used by another guild
{
TGuildNames : : iterator it = _GuildNames . find ( name ) ;
if ( it ! = _GuildNames . end ( ) & & it - > second . GuildId ! = guildId )
return TCharacterNameResult : : cnr_already_exist ;
}
// the name is valid
return TCharacterNameResult : : cnr_ok ;
}
//-----------------------------------------------------------------------------
void CNameManager : : registerLoadedGuildNames ( uint32 shardId , const std : : map < uint32 , ucstring > & guilds , vector < uint32 > & renamedGuildIds )
{
bool saveFile = false ;
std : : map < uint32 , ucstring > : : const_iterator first ( guilds . begin ( ) ) , last ( guilds . end ( ) ) ;
for ( ; first ! = last ; + + first )
{
uint32 guildId = first - > first ;
const ucstring & guildName = first - > second ;
// check the name
if ( isGuildNameUsable ( guildName , guildId ) = = TCharacterNameResult : : cnr_ok )
{
// convert the name into the standard utf8 low case version
string name = toLower ( guildName . toUtf8 ( ) ) ;
// ok, the name is correct, check if it already exist
TGuildNames : : iterator it ( _GuildNames . find ( name ) ) ;
if ( it = = _GuildNames . end ( ) )
{
// this is a new name
_GuildNames . insert ( make_pair ( name , TGuildSlot ( shardId , guildId ) ) ) ;
_GuildIndex . insert ( make_pair ( guildId , name ) ) ;
saveFile = true ;
}
else
{
BOMB_IF ( it - > second . GuildId ! = guildId , " Guild " < < guildId < < " is just loaded by shard " < < shardId < < " but name already in use by guild " < < it - > second . GuildId , continue ) ;
BOMB_IF ( it - > second . ShardId ! = shardId , " Guild " < < guildId < < " is just loaded by shard " < < shardId < < " but already registered by shard " < < it - > second . ShardId , continue ) ;
}
}
else
{
// we need to rename the guild
ucstring newName = generateDefaultGuildName ( guildId ) ;
nlinfo ( " NM:registerLoadedGuildNames : Guild %u has a conflicting name '%s', renamed to '%s' " ,
guildId ,
guildName . toUtf8 ( ) . c_str ( ) ,
newName . toUtf8 ( ) . c_str ( ) ) ;
// save it in the container
std : : string name = toLower ( newName . toUtf8 ( ) ) ;
_GuildNames . insert ( make_pair ( name , TGuildSlot ( shardId , guildId ) ) ) ;
_GuildIndex . insert ( make_pair ( guildId , name ) ) ;
saveFile = true ;
// put it in the renamed guild list return vector
renamedGuildIds . push_back ( guildId ) ;
}
}
if ( saveFile )
{
saveGuildNames ( ) ;
}
}
//-----------------------------------------------------------------------------
bool CNameManager : : assignGuildName ( uint32 shardId , uint32 guildId , const ucstring & guildName )
{
bool ret ;
if ( isGuildNameUsable ( guildName , guildId ) ! = TCharacterNameResult : : cnr_ok )
{
// oups, bad name, we need to generate a new valid name
ucstring newName = generateDefaultGuildName ( guildId ) ;
string name = toLower ( newName . toUtf8 ( ) ) ;
_GuildNames . insert ( make_pair ( name , TGuildSlot ( shardId , guildId ) ) ) ;
_GuildIndex . insert ( make_pair ( guildId , name ) ) ;
ret = false ;
}
else
{
// ok, the name is correct, do a simple insertion
string name = toLower ( guildName . toUtf8 ( ) ) ;
_GuildNames . insert ( make_pair ( name , TGuildSlot ( shardId , guildId ) ) ) ;
_GuildIndex . insert ( make_pair ( guildId , name ) ) ;
ret = true ;
}
// save the file
saveGuildNames ( ) ;
return ret ;
}
//-----------------------------------------------------------------------------
void CNameManager : : releaseGuildName ( uint32 shardId , uint32 guildId )
{
TGuildIndex : : iterator it ( _GuildIndex . find ( guildId ) ) ;
if ( it = = _GuildIndex . end ( ) )
{
nlwarning ( " CNameManager::releaseGuildName : no guild %s registered in name manager " , guildId ) ;
return ;
}
// erase the name entry
_GuildNames . erase ( it - > second ) ;
// erase the index entry
_GuildIndex . erase ( it ) ;
// save the file
saveGuildNames ( ) ;
}
//-----------------------------------------------------------------------------
//void CNameManager::saveCharacterNames()
//{
// // save the character names
// {
// string fileName = "character_names.txt";
//
// CSString s;
// nlinfo("NAMEMGR::save: building file content: %s",fileName.c_str());
// for (TNamesIndex::TAToBMap::const_iterator it=_Names.getAToBMap().begin(); it!=_Names.getAToBMap().end(); ++it)
// {
// s << it->first.Name << " " << it->second.UserId << " " << it->second.CharIndex << " " << it->first.HomeSessionId << "\n";
// }
//
// if( s.empty() )
// return;
//
// nlinfo("NAMEMGR::save: send message to BS");
//
// CBackupMsgSaveFile msg( fileName, CBackupMsgSaveFile::SaveFile, BsiGlobal );
// msg.DataMsg.serialBuffer((uint8*)s.c_str(), s.size());
// BsiGlobal.sendFile(msg);
// }
//
// // save account names
// {
// string fileName = "account_names.txt";
//
// string s;
// nlinfo("NAMEMGR::save: building file content: %s",fileName.c_str());
// for (TAccountNames::iterator it=_AccountNames.begin();it!=_AccountNames.end();++it)
// {
// s+=NLMISC::toString("%s %i\n",(*it).second.c_str(),(*it).first);
// }
//
// if( s.empty() )
// return;
//
// nlinfo("NAMEMGR::save: send message to BS");
//
// CBackupMsgSaveFile msg( fileName, CBackupMsgSaveFile::SaveFile, BsiGlobal );
// msg.DataMsg.serialBuffer((uint8*)s.c_str(), s.size());
// BsiGlobal.sendFile(msg);
// }
//}
//-----------------------------------------------------------------------------
void CNameManager : : saveGuildNames ( )
{
// save the guild names
{
string fileName = GUILD_NAME_FILE ;
string s ;
nlinfo ( " NAMEMGR::save: building file content: %s " , fileName . c_str ( ) ) ;
for ( TGuildNames : : const_iterator it = _GuildNames . begin ( ) ; it ! = _GuildNames . end ( ) ; + + it )
{
s + = NLMISC : : toString ( " %s %i %i \n " , it - > first . c_str ( ) , it - > second . ShardId , it - > second . GuildId ) ;
}
if ( s . empty ( ) )
return ;
nlinfo ( " NAMEMGR::save: send message to BS " ) ;
CBackupMsgSaveFile msg ( fileName , CBackupMsgSaveFile : : SaveFile , Bsi ) ;
2010-05-18 14:53:47 +00:00
msg . DataMsg . serialBuffer ( ( uint8 * ) s . c_str ( ) , ( uint ) s . size ( ) ) ;
2010-05-06 00:08:41 +00:00
Bsi . sendFile ( msg ) ;
}
}
//-----------------------------------------------------------------------------
void CNameManager : : loadAllNames ( )
{
bool result ;
// load account names
result = loadAccountNamesFromDatabase ( ) ;
if ( result = = false )
result = loadAccountNamesFromTxt ( ) ;
if ( result = = false )
nlinfo ( " NAMEMGR::load: Failed to load account names from txt file " ) ;
// load character names
result = loadCharacterNamesFromDatabase ( ) ;
if ( result = = false )
result = loadCharacterNamesFromTxt ( ) ;
// if (result==false)
// result=loadCharacterNamesFromXML();
if ( result = = false )
nlinfo ( " NAMEMGR::load: Failed to load character names from txt file " ) ;
// load character names
result = loadGuildsNamesFromTxt ( ) ;
if ( result = = false )
nlinfo ( " NAMEMGR::load: Failed to load guilds names from txt file " ) ;
// load forbidden names
result = loadForbiddenNames ( ) ;
if ( result = = false )
nlwarning ( " NAMEMGR::load: Failed to load forbidden name file " ) ;
// load reserved names
result = loadReservedNames ( " reserved_names.xml " ) ;
if ( result = = false )
nlwarning ( " NAMEMGR::load: Failed to load reserved player name file " ) ;
// load dev and gm names
result = loadReservedNames ( " dev_gm_names.xml " ) ;
if ( result = = false )
nlwarning ( " NAMEMGR::load: Failed to load reserved dev and gm name file " ) ;
nlinfo ( " NameManager : checking %u loaded names... " , _Names . getAToBMap ( ) . size ( ) ) ;
// revalidate all character names
TNamesIndex : : TAToBMap : : const_iterator first ( _Names . getAToBMap ( ) . begin ( ) ) , last ( _Names . getAToBMap ( ) . end ( ) ) ;
for ( uint i = 0 ; first ! = last ; + + i )
{
TNamesIndex : : TAToBMap : : const_iterator next ( first ) ;
+ + next ;
const TFullName & fullName = first - > first ;
const TCharSlot & charSlot = first - > second ;
if ( isNameUsable ( fullName . Name , charSlot . UserId , charSlot . CharIndex , fullName . HomeSessionId ) ! = CHARSYNC : : TCharacterNameResult : : cnr_ok )
{
nlinfo ( " NameManager : renaming invalid character name '%s' for char %u " , fullName . Name . c_str ( ) , charSlot . getCharId ( ) ) ;
// get the shard id
uint32 charId = charSlot . getCharId ( ) ;
// the name is invalid !
ucstring newName = renameCharacter ( charSlot . getCharId ( ) , fullName . HomeSessionId ) ;
// do not access the fullName and charSlot var from now
RSMGR : : CCharacterPtr character = RSMGR : : CCharacter : : load ( * _Database , charId , __FILE__ , __LINE__ ) ;
if ( character ! = NULL )
{
character - > setCharName ( newName . toUtf8 ( ) ) ;
character - > update ( * _Database ) ;
}
}
first = next ;
if ( i % 1000 = = 0 )
{
nldebug ( " NameManager : checking name advance %u/%u " , i , _Names . getAToBMap ( ) . size ( ) ) ;
}
}
}
//-----------------------------------------------------------------------------
bool CNameManager : : loadAccountNamesFromTxt ( )
{
FILE * f ;
string fileName ;
// open the file
fileName = BsiGlobal . getLocalPath ( ) + " account_names.txt " ;
f = fopen ( fileName . c_str ( ) , " rb " ) ;
if ( f = = NULL )
{
nlwarning ( " Failed to open file for reading: %s " , fileName . c_str ( ) ) ;
return false ;
}
CSString input ;
// read the file content into a buffer
uint32 size = NLMISC : : CFile : : getFileSize ( f ) ;
input . resize ( size ) ;
2010-05-18 14:53:47 +00:00
uint32 readSize = ( uint32 ) fread ( & input [ 0 ] , 1 , size , f ) ;
2010-05-06 00:08:41 +00:00
fclose ( f ) ;
BOMB_IF ( readSize ! = size , " Failed to read file content for file: " + fileName , return false ) ;
// scan the content
while ( ! input . empty ( ) )
{
CSString line = input . firstLine ( true ) ;
if ( line . strip ( ) . empty ( ) )
continue ;
DROP_IF ( line . countWords ( ) ! = 2 , " Invalid line found in account names file: " + line , continue ) ;
DROP_IF ( line . word ( 1 ) . atoi ( ) = = 0 , " Invalid user id in account names file line: " + line , continue ) ;
_AccountNames [ line . word ( 1 ) . atoi ( ) ] = line . word ( 0 ) ;
}
return true ;
}
//-----------------------------------------------------------------------------
bool CNameManager : : loadAccountNamesFromDatabase ( )
{
if ( _Database = = NULL )
return false ;
// request the database to extract the complete list of account names
CSString query ;
query < < " SELECT user_id, user_name FROM ring_users " ;
BOMB_IF ( ! _Database - > query ( query ) , " Failed to load account names from the database " , return false ) ;
auto_ptr < CUseResult > result = _Database - > useResult ( ) ;
while ( result - > fetchRow ( ) )
{
uint32 userId ;
string accountName ;
result - > getField ( 0 , userId ) ;
result - > getField ( 1 , accountName ) ;
_AccountNames [ userId ] = accountName ;
}
return true ;
}
//-----------------------------------------------------------------------------
//bool CNameManager::loadAccountNamesFromXML()
//{
// CIFile f;
// string fileName;
//
// fileName = BsiGlobal.getLocalPath() + "account_names.xml";
// if( !f.open( fileName ) )
// return false;
//
// nlinfo("NAMEMGR: Loading file: %s",fileName.c_str());
//
// CIXml input;
// if(!input.init( f ))
// {
// nlwarning("NAMEMGR:loadAccountNamesFromXML Can't init xml input for file %s", fileName.c_str());
// return false;
// }
// input.serialCont( _AccountNames );
// f.close();
//
// return true;
//}
//-----------------------------------------------------------------------------
//bool CNameManager::loadAccountNamesFromEIDTranslator()
//{
// nlinfo("NAMEMGR: Build account names from EIDTranslator");
//
// try
// {
// CEntityIdTranslator::getInstance()->load(BsiGlobal.getLocalPath() + "eid_translation.data", IService::getInstance()->ConfigFile.getVar("InvalidEntityNamesFilename").asString());
// }
// catch(Exception &)
// {
// // if we can't load the file, we force a check coherency
// nlwarning("Can't load the eid_translation.data");
// return false;
// }
//
// for( map<CEntityId, CEntityIdTranslator::CEntity>::const_iterator it = CEntityIdTranslator::getInstance()->getRegisteredEntities().begin(); it != CEntityIdTranslator::getInstance()->getRegisteredEntities().end(); ++it )
// {
// _AccountNames[(*it).second.UId]= (*it).second.UserName;
// }
// saveCharacterNames();
//
// return true;
//}
//-----------------------------------------------------------------------------
bool CNameManager : : loadCharacterNamesFromTxt ( )
{
FILE * f ;
string fileName ;
// open the file
fileName = BsiGlobal . getLocalPath ( ) + " character_names.txt " ;
f = fopen ( fileName . c_str ( ) , " rb " ) ;
if ( f = = NULL )
{
nlwarning ( " Failed to open file for reading: %s " , fileName . c_str ( ) ) ;
return false ;
}
CSString input ;
// read the file content into a buffer
uint32 size = NLMISC : : CFile : : getFileSize ( f ) ;
input . resize ( size ) ;
2010-05-18 14:53:47 +00:00
uint32 readSize = ( uint32 ) fread ( & input [ 0 ] , 1 , size , f ) ;
2010-05-06 00:08:41 +00:00
fclose ( f ) ;
BOMB_IF ( readSize ! = size , " Failed to read file content for file: " + fileName , return false ) ;
_Names . clear ( ) ;
_DuplicatedSlots . clear ( ) ;
typedef map < TCharSlot , TName > TCharSlotToName ;
TCharSlotToName charSlotToName ;
vector < string > lines ;
NLMISC : : explode ( string ( input ) , string ( " \n " ) , lines , true ) ;
// scan the content
// while (!input.empty())
// bool mustSaveFile = false;
for ( uint i = 0 ; i < lines . size ( ) ; + + i )
{
if ( i > 0 & & ( i % 100 = = 0 ) )
nldebug ( " Loading character names : %u/%u (%.2f%%) " , i , lines . size ( ) , float ( i ) / float ( lines . size ( ) ) * 100.0f ) ;
// CSString line=input.firstLine(true);
CSString line = lines [ i ] ;
vector < string > words ;
NLMISC : : explode ( string ( line ) , string ( " " ) , words , true ) ;
if ( words . empty ( ) )
continue ;
// if (line.strip().empty())
// continue;
// BOMB_IF (line.countWords()!=3,"Invalid line found in character names file: "+line,continue);
BOMB_IF ( words . size ( ) ! = 3 & & words . size ( ) ! = 4 , " Invalid line found in character names file: " + line , continue ) ;
int i1 = atoi ( words [ 1 ] . c_str ( ) ) ;
int i2 = atoi ( words [ 2 ] . c_str ( ) ) ;
BOMB_IF ( i1 = = 0 , " Invalid user id in character names file line: " + line , continue ) ;
BOMB_IF ( i2 > 15 | | ( i2 = = 0 & & words [ 2 ] ! = " 0 " ) , " Invalid slot id in character names file line: " + line , continue ) ;
int sessionId = 0 ;
if ( words . size ( ) > 3 )
sessionId = atoi ( words [ 3 ] . c_str ( ) ) ;
const TName name = words [ 0 ] ;
const TCharSlot charSlot = TCharSlot ( i1 , i2 ) ;
TFullName fullname ( name , sessionId ) ;
// Check the name is usable, otherwise skip (to resolve corrupted names)
if ( isNameUsable ( ucstring ( name ) , charSlot . UserId , charSlot . CharIndex , sessionId ) ! = TCharacterNameResult : : cnr_ok )
{
nlwarning ( " Invalid character name '%s' for user %u char slot %u in %s, will be reset to default " , name . c_str ( ) , charSlot . UserId , ( uint ) charSlot . CharIndex , fileName . c_str ( ) ) ;
// mustSaveFile = true;
continue ;
}
if ( _Names . getAToBMap ( ) . find ( fullname ) ! = _Names . getAToBMap ( ) . end ( ) )
_Names . removeWithA ( fullname ) ;
if ( _Names . getBToAMap ( ) . find ( charSlot ) ! = _Names . getBToAMap ( ) . end ( ) )
_Names . removeWithB ( charSlot ) ;
_Names . add ( fullname , charSlot ) ;
// TNames::const_iterator it = _Names.find( name );
// nlassert(it != _Names.end());
// check duplicated character slots
TCharSlotToName : : iterator itSlot = charSlotToName . find ( charSlot ) ;
if ( itSlot ! = charSlotToName . end ( ) )
{
_DuplicatedSlots [ charSlot ] . insert ( ( * itSlot ) . second ) ;
_DuplicatedSlots [ charSlot ] . insert ( name ) ;
}
else
{
charSlotToName [ charSlot ] = name ;
}
// // update the EID translator
// uint64 lid = ( (*it).second.UserId<<4 ) | (*it).second.CharId;
// CEntityId id( RYZOMID::player, lid );
// id.setDynamicId( 0 );
// id.setCreatorId( 0 );
//
// if (CEntityIdTranslator::getInstance()->isEntityRegistered(id))
// CEntityIdTranslator::getInstance()->unregisterEntity( id );
// CEntityIdTranslator::getInstance()->registerEntity( id, capitalize( (*it).first ), (*it).second.CharId, (*it).second.UserId, _AccountNames[(*it).second.UserId] );
}
if ( ! _DuplicatedSlots . empty ( ) )
{
nlwarning ( " NAMEMGR: file '%s' contains %u duplicated character slots " , fileName . c_str ( ) , _DuplicatedSlots . size ( ) ) ;
}
// if ( mustSaveFile )
// {
// saveCharacterNames();
// }
return true ;
}
bool CNameManager : : loadCharacterNamesFromDatabase ( )
{
if ( _Database = = NULL )
return false ;
_Names . clear ( ) ;
// request the database to extract the complete list of character names
CSString query ;
query < < " SELECT char_id, char_name, home_mainland_session_id FROM characters " ;
BOMB_IF ( ! _Database - > query ( query ) , " Failed to load character names from the database " , return false ) ;
auto_ptr < CUseResult > result = _Database - > useResult ( ) ;
while ( result - > fetchRow ( ) )
{
uint32 charId ;
string charName ;
uint32 sessionNum ;
result - > getField ( 0 , charId ) ;
result - > getField ( 1 , charName ) ;
result - > getField ( 2 , sessionNum ) ;
// uint32 userId = charId >> 4;
// uint8 charSlot = uint8(charId & 0xf);
_Names . add ( TFullName ( charName , sessionNum ) , TCharSlot ( charId ) ) ;
}
return true ;
}
//-----------------------------------------------------------------------------
//bool CNameManager::loadCharacterNamesFromXML()
//{
// CIFile f;
// string fileName;
//
// // load the character names and insert them into the eid translator
// fileName = BsiGlobal.getLocalPath() + "character_names.xml";
// if( !f.open( fileName ) )
// return false;
//
// nlinfo("NAMEMGR: Loading file: %s",fileName.c_str());
// CIXml input;
// if(!input.init( f ))
// {
// nlwarning("<NAMEMGR::loadCharacterNamesFromXML>Can't init xml input for file %s", fileName.c_str());
// return false;
// }
// TNames namesToCheckAndAdd;
// input.serialCont( namesToCheckAndAdd );
// f.close();
//
// _DuplicatedSlots.clear();
// map<TCharSlot,TName> charSlotToName;
//
// // update the EID translator
// bool mustSaveFile = false;
// for( TNames::iterator it = namesToCheckAndAdd.begin(); it != namesToCheckAndAdd.end(); ++it )
// {
// const TName & name = (*it).first;
// const TCharSlot charSlot = TCharSlot( (*it).second.UserId, (*it).second.CharIndex );
//
// // Check the name is usable, otherwise skip (to resolve corrupted names)
// if ( isNameUsable( ucstring(name), charSlot.UserId, charSlot.CharIndex ) != TCharacterNameResult::cnr_ok)
// {
// nlwarning( "Invalid character name '%s' for user %u char slot %u in %s, will be reset to default", name.c_str(), charSlot.UserId, (uint)charSlot.CharIndex, fileName.c_str() );
// mustSaveFile = true;
// continue;
// }
//
// if (_Names.getAToBMap().find(it->first) != _Names.getAToBMap().end())
// _Names.removeWithA(it->first);
// if (_Names.getBToAMap().find(it->second) != _Names.getBToAMap().end())
// _Names.removeWithB(it->second);
//
// _Names.add( it->first, it->second );
//
// // check duplicated character slots
// map<TCharSlot,TName>::iterator itSlot = charSlotToName.find( charSlot );
// if ( itSlot != charSlotToName.end() )
// {
// _DuplicatedSlots[charSlot].insert( (*itSlot).second );
// _DuplicatedSlots[charSlot].insert( name );
// }
// else
// {
// charSlotToName[charSlot] = name;
// }
//
//// uint64 lid = ( (*it).second.UserId<<4 ) | (*it).second.CharId;
//// CEntityId id( RYZOMID::player, lid );
//// id.setDynamicId( 0 );
//// id.setCreatorId( 0 );
////
//// if (CEntityIdTranslator::getInstance()->isEntityRegistered(id))
//// CEntityIdTranslator::getInstance()->unregisterEntity( id );
//// CEntityIdTranslator::getInstance()->registerEntity( id, capitalize( name ), (*it).second.CharId, (*it).second.UserId, _AccountNames[(*it).second.UserId] );
// }
//
// if ( !_DuplicatedSlots.empty() )
// {
// nlwarning("NAMEMGR: file '%s' contains %u duplicated character slots", fileName.c_str(), _DuplicatedSlots.size() );
// }
//
// if ( mustSaveFile )
// {
// saveCharacterNames();
// }
//
// return true;
//}
//-----------------------------------------------------------------------------
bool CNameManager : : loadGuildsNamesFromTxt ( )
{
FILE * f ;
string fileName ;
// open the file
fileName = Bsi . getLocalPath ( ) + GUILD_NAME_FILE ;
f = fopen ( fileName . c_str ( ) , " rb " ) ;
if ( f = = NULL )
{
nlinfo ( " Failed to open file for reading: %s " , fileName . c_str ( ) ) ;
return false ;
}
CSString input ;
// read the file content into a buffer
uint32 size = NLMISC : : CFile : : getFileSize ( f ) ;
input . resize ( size ) ;
2010-05-18 14:53:47 +00:00
uint32 readSize = ( uint32 ) fread ( & input [ 0 ] , 1 , size , f ) ;
2010-05-06 00:08:41 +00:00
fclose ( f ) ;
BOMB_IF ( readSize ! = size , " Failed to read file content for file: " + fileName , return false ) ;
_GuildNames . clear ( ) ;
_GuildIndex . clear ( ) ;
typedef map < TGuildSlot , TName > TGuildSlotToName ;
TGuildSlotToName guildSlotToName ;
vector < string > lines ;
NLMISC : : explode ( string ( input ) , string ( " \n " ) , lines , true ) ;
// scan the content
// while (!input.empty())
// bool mustSaveFile = false;
for ( uint i = 0 ; i < lines . size ( ) ; + + i )
{
if ( i > 0 & & ( i % 100 = = 0 ) )
nldebug ( " Loading guild names : %u/%u (%.2f%%) " , i + 1 , lines . size ( ) , float ( i + 1 ) / float ( lines . size ( ) ) * 100.0f ) ;
CSString line = lines [ i ] ;
vector < string > words ;
NLMISC : : explode ( string ( line ) , string ( " " ) , words , true ) ;
if ( words . empty ( ) )
continue ;
// merge the first words until we have only 3 words
while ( words . size ( ) > 3 )
{
words [ 0 ] + = words [ 1 ] ;
words . erase ( words . begin ( ) + 1 ) ;
}
BOMB_IF ( words . size ( ) ! = 3 , " Invalid line " < < i + 1 < < " found in guild names file : ' " < < line < < " ' " , continue ) ;
int i1 = atoi ( words [ 1 ] . c_str ( ) ) ;
int i2 = atoi ( words [ 2 ] . c_str ( ) ) ;
BOMB_IF ( i1 = = 0 , " Invalid shardId in guild names file line " < < i + 1 < < " : ' " < < line < < " ' " , continue ) ;
BOMB_IF ( i2 = = 0 , " Invalid guildId in guild names file line " < < i + 1 < < " : ' " < < line < < " ' " , continue ) ;
TName name = words [ 0 ] ;
const TGuildSlot guildSlot = TGuildSlot ( i1 , i2 ) ;
// Check the name is usable, i.e valid regarding guild name rules AND not already used
if ( isGuildNameUsable ( ucstring ( name ) , guildSlot . GuildId ) ! = TCharacterNameResult : : cnr_ok )
{
nlwarning ( " Invalid guild name '%s' for guild %u on from shard %u will be reset to default " , name . c_str ( ) , guildSlot . GuildId , guildSlot . ShardId ) ;
name = toLower ( generateDefaultGuildName ( guildSlot . GuildId ) . toUtf8 ( ) ) ;
}
_GuildNames . insert ( make_pair ( name , guildSlot ) ) ;
_GuildIndex . insert ( make_pair ( guildSlot . GuildId , name ) ) ;
}
// if (mustSaveFile)
// {
// saveGuildNames();
// }
return true ;
}
//-----------------------------------------------------------------------------
bool CNameManager : : loadForbiddenNames ( )
{
string fileName ;
// read the invalid entity name list
fileName = IService : : getInstance ( ) - > WriteFilesDirectory . toString ( ) + " invalid_entity_names.txt " ;
FILE * fp = fopen ( fileName . c_str ( ) , " r " ) ;
if ( fp = = NULL )
return false ;
nlinfo ( " NAMEMGR: Loading file: %s " , fileName . c_str ( ) ) ;
while ( true )
{
char str [ 512 ] ;
fgets ( str , 511 , fp ) ;
if ( feof ( fp ) )
break ;
if ( strlen ( str ) > 0 )
{
str [ strlen ( str ) - 1 ] = ' \0 ' ;
toLower ( str ) ;
_ForbiddenNames . push_back ( string ( ) ) ;
_ForbiddenNames . back ( ) = str ;
}
}
fclose ( fp ) ;
return true ;
}
//-----------------------------------------------------------------------------
bool CNameManager : : loadReservedNames ( const char * fileNameWithoutPath )
{
CIFile f ;
string fileName ;
// read the reserved name list
fileName = IService : : getInstance ( ) - > WriteFilesDirectory . toString ( ) + fileNameWithoutPath ;
if ( ! f . open ( fileName ) )
return false ;
nlinfo ( " NAMEMGR: Loading file: %s " , fileName . c_str ( ) ) ;
CIXml input ;
if ( ! input . init ( f ) )
{
nlwarning ( " <NAMEMGR::loadReservedNames>Can't init xml input for file %s " , fileName . c_str ( ) ) ;
return false ;
}
TReservedNames reservedName ;
input . serialCont ( reservedName ) ;
f . close ( ) ;
for ( TReservedNames : : iterator it = reservedName . begin ( ) ; it ! = reservedName . end ( ) ; + + it )
{
_ReservedNames [ toLower ( ( * it ) . first ) ] = ( * it ) . second ;
}
return true ;
}
void CNameManager : : update ( )
{
TTempReservedNames : : iterator first ( _TemporaryReservedNames . begin ( ) ) , last ( _TemporaryReservedNames . end ( ) ) ;
uint32 now = CTime : : getSecondsSince1970 ( ) ;
for ( ; first ! = last ; + + first )
{
if ( first - > second . ReserveDate + TEMPORARY_RESERVED_NAME_EXPIRATION > now )
{
// this reservation has expired, release it
_TemporaryReservedNames . erase ( first ) ;
// next release at next update
break ;
}
}
}
const CNameManager : : TName & CNameManager : : getGuildName ( uint32 guildId ) const
{
TGuildIndex : : const_iterator it ( _GuildIndex . find ( guildId ) ) ;
if ( it = = _GuildIndex . end ( ) )
{
static string emptyString ;
return emptyString ;
}
else
return it - > second ;
}
uint32 CNameManager : : findCharId ( const std : : string & charName , uint32 homeSessionId )
{
TFullName fullname ( charName , homeSessionId ) ;
TNamesIndex : : TAToBMap : : const_iterator it ( _Names . getAToBMap ( ) . find ( fullname ) ) ;
if ( it ! = _Names . getAToBMap ( ) . end ( ) )
{
return it - > second . getCharId ( ) ;
}
// not found
return 0 ;
}
/// Try to find a shard id from a name without session id, return 0 if 0 or more than one match.
uint32 CNameManager : : findCharId ( const std : : string & charName )
{
TFullName fullname ( charName , 0 ) ;
TNamesIndex : : TAToBMap : : const_iterator it ( _Names . getAToBMap ( ) . upper_bound ( fullname ) ) ;
TNamesIndex : : TAToBMap : : const_iterator found ( _Names . getAToBMap ( ) . end ( ) ) ;
while ( it - > first . Name = = charName )
{
if ( found ! = _Names . getAToBMap ( ) . end ( ) )
{
// more than one match
return 0 ;
}
// first found
found = it ;
}
if ( it ! = _Names . getAToBMap ( ) . end ( ) )
{
// we found one
return it - > second . getCharId ( ) ;
}
// not found 0
return 0 ;
}
NLMISC_CLASS_COMMAND_IMPL ( CNameManager , dump )
{
if ( args . size ( ) = = 1 & & args [ 0 ] = = " all " )
{
// dump the complete name table (heavy !)
const TNamesIndex : : TBToAMap & ba = _Names . getBToAMap ( ) ;
log . displayNL ( " Dumping %u character names : " , ba . size ( ) ) ;
TNamesIndex : : TBToAMap : : const_iterator first ( ba . begin ( ) ) , last ( ba . end ( ) ) ;
for ( ; first ! = last ; + + first )
{
const TCharSlot & charSlot = first - > first ;
const TFullName & fullName = first - > second ;
log . displayNL ( " User %u, Slot %u (char_id %u) : Name '%s' on session %u " ,
charSlot . UserId ,
charSlot . CharIndex ,
charSlot . UserId * 16 + charSlot . CharIndex ,
fullName . Name . c_str ( ) ,
fullName . HomeSessionId . asInt ( ) ) ;
}
return true ;
}
if ( args . size ( ) ! = 0 )
return false ;
log . displayNL ( " Dumping NameManager internal state : " ) ;
log . displayNL ( " - %u character names " , _Names . getAToBMap ( ) . size ( ) ) ;
log . displayNL ( " - %u account names " , _AccountNames . size ( ) ) ;
log . displayNL ( " - %u guild names " , _GuildNames . size ( ) ) ;
log . displayNL ( " - %u guild index " , _GuildIndex . size ( ) ) ;
log . displayNL ( " - %u reserved names " , _ReservedNames . size ( ) ) ;
log . displayNL ( " - %u forbidden names " , _ForbiddenNames . size ( ) ) ;
log . displayNL ( " End of dump " ) ;
return true ;
}
NLMISC_CLASS_COMMAND_IMPL ( CNameManager , releaseGuildNamesForShard )
{
if ( args . size ( ) ! = 1 )
return false ;
uint32 shardId = atoui ( args [ 1 ] . c_str ( ) ) ;
vector < uint32 > guildNameToRemove ;
TGuildNames : : iterator first ( _GuildNames . begin ( ) ) , last ( _GuildNames . end ( ) ) ;
for ( ; first ! = last ; + + first )
{
if ( first - > second . ShardId = = shardId )
guildNameToRemove . push_back ( first - > second . GuildId ) ;
}
log . displayNL ( " Releasing %u guild name for shard %u " , guildNameToRemove . size ( ) , shardId ) ;
for ( uint i = 0 ; i < guildNameToRemove . size ( ) ; + + i )
{
uint32 guildId = guildNameToRemove [ i ] ;
TGuildIndex : : iterator it ( _GuildIndex . find ( guildId ) ) ;
BOMB_IF ( it = = _GuildIndex . end ( ) , " Error in guildname/index management " , continue ) ;
_GuildNames . erase ( it - > second ) ;
_GuildIndex . erase ( it ) ;
}
return true ;
}