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 "../pd_lib/pds_common.h"
# include "pds_database.h"
# include "pds_table.h"
# include "pds_type.h"
# include "database_adapter.h"
# include "db_manager.h"
# include "pd_lib/pd_server_utils.h"
# include "pd_lib/reference_builder.h"
# include "pd_lib/delta_builder.h"
# include <nel/misc/debug.h>
# include <nel/misc/path.h>
# include <nel/misc/config_file.h>
# include <nel/misc/file.h>
# include <nel/misc/variable.h>
# include <nel/misc/o_xml.h>
# include <nel/misc/i_xml.h>
# include <nel/net/service.h>
using namespace std ;
using namespace NLMISC ;
using namespace NLNET ;
// Hosted by db_manager.cpp
extern NLMISC : : CVariable < uint > DeltaUpdateRate ;
/*
* Constructor
*/
CDatabase : : CDatabase ( uint32 id ) : _Reference ( id )
{
clear ( ) ;
_State . Id = id ;
}
/*
* Destructor
*/
CDatabase : : ~ CDatabase ( )
{
PDS_FULL_DEBUG ( " delete() " ) ;
clear ( ) ;
}
/*
* Massive Database clear
*/
void CDatabase : : clear ( )
{
uint i ;
for ( i = 0 ; i < _Types . size ( ) ; + + i )
{
if ( _Types [ i ] ! = NULL )
delete _Types [ i ] ;
_Types [ i ] = NULL ;
}
_Types . clear ( ) ;
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
{
if ( _Tables [ i ] ! = NULL )
delete _Tables [ i ] ;
_Tables [ i ] = NULL ;
}
_Tables . clear ( ) ;
_Init = false ;
//_ObjectList.clear();
_SetMap . clear ( ) ;
_State . Name . clear ( ) ;
_ServiceId . set ( 0xffff ) ;
}
/*
* Init database
* \ param xmlStream is the xml description of the database
*/
bool CDatabase : : init ( )
{
PDS_DEBUG ( " init() " ) ;
// clear all before doing anything
clear ( ) ;
const CDatabaseNode & db = _Description . getDatabaseNode ( ) ;
_State . Name = db . Name ;
uint i ;
for ( i = 0 ; i < db . Types . size ( ) ; + + i )
{
CType * type = new CType ;
if ( ! type - > init ( this , db . Types [ i ] ) )
{
PDS_WARNING ( " init(): failed to init type '%d' " , i ) ;
return false ;
}
if ( type - > getId ( ) ! = i )
{
PDS_WARNING ( " init(): failed to init type '%d' " , i ) ;
return false ;
}
_Types . push_back ( type ) ;
}
for ( i = 0 ; i < db . Tables . size ( ) ; + + i )
{
CTable * table = new CTable ( ) ;
if ( ! table - > init ( this , db . Tables [ i ] ) | | table - > getId ( ) ! = i )
{
PDS_WARNING ( " init(): failed to init table '%d' " , i ) ;
return false ;
}
_Tables . push_back ( table ) ;
}
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
{
CTable * table = _Tables [ i ] ;
if ( ! table - > buildColumns ( ) )
{
PDS_WARNING ( " init(): failed to build table '%d' columns " , i ) ;
return false ;
}
}
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
{
_Tables [ i ] - > postInit ( ) ;
}
_Init = true ;
if ( ! notifyNewReference ( false ) )
{
_Init = false ;
return false ;
}
// save description in log dir
std : : string logDir = RY_PDS : : CPDSLib : : getLogDirectory ( _State . Id ) ;
if ( ! CFile : : isExists ( logDir ) | | ! CFile : : isDirectory ( logDir ) )
{
if ( ! CFile : : createDirectoryTree ( logDir ) )
{
PDS_WARNING ( " init(): failed to create log root directory '%s' " , logDir . c_str ( ) ) ;
}
if ( ! CFile : : setRWAccess ( logDir ) )
{
PDS_WARNING ( " init(): failed, can't set RW access to directory '%s' " , logDir . c_str ( ) ) ;
}
}
// CTimestamp initDate;
// initDate.setToCurrent();
// _Description.saveDescription(logDir + initDate.toString() + ".description");
PDS_DEBUG ( " init() successful " ) ;
return true ;
}
/*
* Checkup database
*/
bool CDatabase : : checkup ( )
{
PDS_DEBUG ( " checkup() " ) ;
if ( ! initialised ( ) )
{
PDS_WARNING ( " checkup(): database not initialised " ) ;
return false ;
}
uint i ;
bool dbOk = true ;
for ( i = 0 ; i < _Types . size ( ) ; + + i )
{
if ( _Types [ i ] = = NULL | | ! _Types [ i ] - > initialised ( ) )
{
PDS_WARNING ( " checkup(): type '%d' not initialised " , i ) ;
continue ;
}
}
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
{
if ( _Tables [ i ] = = NULL | | ! _Tables [ i ] - > initialised ( ) )
{
PDS_WARNING ( " checkup(): table '%d' not initialised " , i ) ;
continue ;
}
CTable * table = _Tables [ i ] ;
const std : : vector < CColumn > & columns = table - > getColumns ( ) ;
const std : : vector < CAttribute * > & attributes = table - > getAttributes ( ) ;
uint j ;
// walk through attributes
for ( j = 0 ; j < attributes . size ( ) ; + + j )
{
if ( attributes [ j ] = = NULL | | ! attributes [ j ] - > initialised ( ) )
{
PDS_WARNING ( " checkup(): attribute '%s:%d' not initialised " , table - > getName ( ) . c_str ( ) , i ) ;
dbOk = false ;
continue ;
}
const CAttribute * attribute = attributes [ j ] ;
if ( attribute - > getParent ( ) ! = table )
{
PDS_WARNING ( " checkup(): attribute '%s:%s' doesn't point table '%d' " , table - > getName ( ) . c_str ( ) , attribute - > getName ( ) . c_str ( ) , i ) ;
dbOk = false ;
}
uint k ;
for ( k = attribute - > getOffset ( ) ; k < attribute - > getOffset ( ) + attribute - > getColumns ( ) ; + + k )
{
if ( k > = columns . size ( ) | | ! columns [ k ] . initialised ( ) )
{
PDS_WARNING ( " checkup(): attribute '%s:%s' points to not initialised column '%d' " , table - > getName ( ) . c_str ( ) , attribute - > getName ( ) . c_str ( ) , k ) ;
dbOk = false ;
}
else if ( columns [ k ] . getParent ( ) ! = attribute )
{
PDS_WARNING ( " checkup(): attribute '%s:%s' points to column '%s' that doesn't point back " , table - > getName ( ) . c_str ( ) , attribute - > getName ( ) . c_str ( ) , columns [ k ] . getName ( ) . c_str ( ) ) ;
dbOk = false ;
}
else
{
const CColumn & column = columns [ k ] ;
}
}
}
// walk through columns
for ( j = 0 ; j < columns . size ( ) ; + + j )
{
if ( ! columns [ j ] . initialised ( ) )
{
PDS_WARNING ( " checkup(): column '%s:%d' not initialised " , table - > getName ( ) . c_str ( ) , j ) ;
dbOk = false ;
continue ;
}
}
}
if ( dbOk )
{
PDS_DEBUG ( " checkup(): Database is correct " ) ;
}
PDS_DEBUG ( " checkup() finished " ) ;
return true ;
}
/*
* Initialise internal timestamps
*/
void CDatabase : : initTimestamps ( )
{
// initialise update and creation timestamps to now
_CreationTimestamp . setToCurrent ( ) ;
_MinuteUpdateTimestamp = _CreationTimestamp ;
_HourUpdateTimestamp = _CreationTimestamp ;
_DayUpdateTimestamp = _CreationTimestamp ;
}
/*
* Get value as a string
* \ param path is of the form ' [ tableindex | tablename ] . [ $ key | index ] . attrib1 . attrib2 '
*/
string CDatabase : : getValue ( const CLocatePath : : TLocatePath & path )
{
if ( path . size ( ) < 3 )
{
PDS_WARNING ( " getValue(): path is too short " ) ;
return " " ;
}
uint node = 0 ;
// select table
const CTable * table ;
uint tableId ;
if ( sscanf ( path [ node ] . Name . c_str ( ) , " %d " , & tableId ) = = 1 )
table = getTable ( tableId ) ;
else
table = getTable ( path [ node ] . Name ) ;
if ( table = = NULL )
{
PDS_WARNING ( " getValue(): unable to select table '%s' " , path [ node ] . Name . c_str ( ) ) ;
return " " ;
}
+ + node ;
// select row
RY_PDS : : CObjectIndex object ;
if ( path [ node ] . Name [ 0 ] = = ' $ ' )
{
uint64 key ;
if ( sscanf ( path [ node ] . Name . c_str ( ) + 1 , " % " NL_I64 " X " , & key ) ! = 1 )
{
PDS_WARNING ( " getValue(): unable to select mapped row '%s' " , path [ node ] . Name . c_str ( ) ) ;
return " " ;
}
object = table - > getMappedRow ( key ) ;
}
else
{
RY_PDS : : TRowIndex row ;
if ( sscanf ( path [ node ] . Name . c_str ( ) , " %u " , & row ) ! = 1 )
{
PDS_WARNING ( " getValue(): unable to select row '%s' " , path [ node ] . Name . c_str ( ) ) ;
return " " ;
}
object = RY_PDS : : CObjectIndex ( ( RY_PDS : : TTableIndex ) table - > getId ( ) , row ) ;
}
if ( ! object . isValid ( ) | | ! isAllocated ( object ) )
{
PDS_WARNING ( " getValue(): object '%s' not accessible " , path [ node ] . Name . c_str ( ) ) ;
return " " ;
}
+ + node ;
const CTable * subTable = table ;
// browse through row
while ( node < path . size ( ) )
{
const CAttribute * attribute = subTable - > getAttribute ( path [ node ] . Name ) ;
if ( attribute = = NULL )
{
PDS_WARNING ( " getValue(): '%s' is not an attribute of '%s' " , path [ node ] . Name . c_str ( ) , subTable - > getName ( ) . c_str ( ) ) ;
return " " ;
}
+ + node ;
}
return " " ;
}
/*
* Get Table
*/
const CTable * CDatabase : : getTable ( const std : : string & name ) const
{
uint i ;
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
if ( _Tables [ i ] ! = NULL & & _Tables [ i ] - > getName ( ) = = name )
return _Tables [ i ] ;
return NULL ;
}
/*
* Get Type
*/
const CType * CDatabase : : getType ( const std : : string & name ) const
{
uint i ;
for ( i = 0 ; i < _Types . size ( ) ; + + i )
if ( _Types [ i ] ! = NULL & & _Types [ i ] - > getName ( ) = = name )
return _Types [ i ] ;
return NULL ;
}
/*
* Get Attribute
*/
const CAttribute * CDatabase : : getAttribute ( uint32 tableId , uint32 attributeId ) const
{
const CTable * table = getTable ( tableId ) ;
if ( table = = NULL )
return NULL ;
return table - > getAttribute ( attributeId ) ;
}
/*
* Get Column
*/
const CColumn * CDatabase : : getColumn ( uint32 tableId , uint32 columnId ) const
{
const CTable * table = getTable ( tableId ) ;
if ( table = = NULL )
return NULL ;
return table - > getColumn ( columnId ) ;
}
/*
* Allocate a row in a table
* \ param index is the table / row to allocate
* Return true if succeded
*/
bool CDatabase : : allocate ( const RY_PDS : : CObjectIndex & index )
{
H_AUTO ( PDS_Database_allocate ) ;
if ( ! initialised ( ) )
{
PDS_WARNING ( " allocate(): database not initialised " ) ;
return false ;
}
if ( ! index . isValid ( ) )
{
PDS_WARNING ( " allocate(): index '%s' is not valid " , index . toString ( ) . c_str ( ) ) ;
return false ;
}
CTable * table = getNonConstTable ( index . table ( ) ) ;
if ( table = = NULL | | ! table - > initialised ( ) )
{
PDS_WARNING ( " allocate(): table '%d' is not initialised " , index . table ( ) ) ;
return false ;
}
bool success = table - > allocate ( index . row ( ) ) ;
if ( success )
PDS_FULL_DEBUG ( " allocated '%s' successfully " , index . toString ( ) . c_str ( ) ) ;
return success ;
}
/*
* Deallocate a row in a table
* \ param index is the table / row to deallocate
* Return true if succeded
*/
bool CDatabase : : deallocate ( const RY_PDS : : CObjectIndex & index )
{
H_AUTO ( PDS_Database_deallocate ) ;
if ( ! initialised ( ) )
{
PDS_WARNING ( " deallocate(): database not initialised " ) ;
return false ;
}
if ( ! index . isValid ( ) )
{
PDS_WARNING ( " deallocate(): index '%s' is not valid " , index . toString ( ) . c_str ( ) ) ;
return false ;
}
CTable * table = getNonConstTable ( index . table ( ) ) ;
if ( table = = NULL | | ! table - > initialised ( ) )
{
PDS_WARNING ( " deallocate(): table '%d' is not initialised " , index . table ( ) ) ;
return false ;
}
bool success = table - > deallocate ( index . row ( ) ) ;
if ( success )
PDS_FULL_DEBUG ( " deallocated '%s' successfully " , index . toString ( ) . c_str ( ) ) ;
return success ;
}
/*
* Tells if an object is allocated
* \ param object is the object index to test
*/
bool CDatabase : : isAllocated ( const RY_PDS : : CObjectIndex & index ) const
{
const CTable * table = getTable ( index . table ( ) ) ;
return table ! = NULL & & table - > isAllocated ( index . row ( ) ) ;
}
/*
* Map a row in a table
* \ param index is the table / row to allocate
* \ param key is the 64 bits row key
* Return true if succeded
*/
bool CDatabase : : mapRow ( const RY_PDS : : CObjectIndex & index , uint64 key )
{
if ( ! initialised ( ) )
{
PDS_WARNING ( " mapRow(): database not initialised " ) ;
return false ;
}
if ( ! index . isValid ( ) )
{
PDS_WARNING ( " mapRow(): index '%s' is not valid " , index . toString ( ) . c_str ( ) ) ;
return false ;
}
CTable * table = getNonConstTable ( index . table ( ) ) ;
if ( table = = NULL | | ! table - > initialised ( ) )
{
PDS_WARNING ( " deallocate(): table '%d' is not initialised " , index . table ( ) ) ;
return false ;
}
bool success = table - > mapRow ( index , key ) ;
if ( success )
PDS_FULL_DEBUG ( " mapped '%016 " NL_I64 " X' to '%s' successfully " , key , index . toString ( ) . c_str ( ) ) ;
return success ;
}
/*
* Unmap a row in a table
* \ param tableIndex is the table to find row
* \ param key is the 64 bits row key
* Return true if succeded
*/
bool CDatabase : : unmapRow ( RY_PDS : : TTableIndex tableIndex , uint64 key )
{
if ( ! initialised ( ) )
{
PDS_WARNING ( " unmapRow(): database not initialised " ) ;
return false ;
}
CTable * table = getNonConstTable ( tableIndex ) ;
if ( table = = NULL | | ! table - > initialised ( ) )
{
PDS_WARNING ( " deallocate(): table '%d' is not initialised " , tableIndex ) ;
return false ;
}
bool success = table - > unmapRow ( key ) ;
if ( success )
PDS_FULL_DEBUG ( " unmapped '%016 " NL_I64 " X' successfully " , key ) ;
return success ;
}
/**
* Get a mapped row
* \ param tableIndex is the table in which the row is mapped
* \ param key is the 64 bits row key
* Return a valid CObjectIndex if success
*/
RY_PDS : : CObjectIndex CDatabase : : getMappedRow ( RY_PDS : : TTableIndex tableIndex , uint64 key ) const
{
if ( ! initialised ( ) )
{
PDS_WARNING ( " getMappedRow(): database not initialised " ) ;
return RY_PDS : : CObjectIndex ( ) ;
}
const CTable * table = getTable ( tableIndex ) ;
if ( table = = NULL | | ! table - > initialised ( ) )
{
PDS_WARNING ( " getMappedRow(): table '%d' is not initialised " , tableIndex ) ;
return RY_PDS : : CObjectIndex ( ) ;
}
return table - > getMappedRow ( key ) ;
}
/**
* Release a row in a table
* \ param index is the table / row to release
* Return true if succeded
*/
bool CDatabase : : release ( const RY_PDS : : CObjectIndex & index )
{
if ( ! initialised ( ) )
{
PDS_WARNING ( " release(): database not initialised " ) ;
return false ;
}
if ( ! index . isValid ( ) )
{
PDS_WARNING ( " release(): index '%s' is not valid " , index . toString ( ) . c_str ( ) ) ;
return false ;
}
CTable * table = getNonConstTable ( index . table ( ) ) ;
if ( table = = NULL | | ! table - > initialised ( ) )
{
PDS_WARNING ( " release(): table '%d' is not initialised " , index . table ( ) ) ;
return false ;
}
bool success = table - > release ( index . row ( ) ) ;
if ( success )
PDS_FULL_DEBUG ( " released '%s' successfully " , index . toString ( ) . c_str ( ) ) ;
return success ;
}
/*
* Release all rows in all table
* Typically , the client disconnected , there is no need to keep rows
*/
bool CDatabase : : releaseAll ( )
{
if ( ! initialised ( ) )
{
PDS_WARNING ( " releaseAll(): database not initialised " ) ;
return false ;
}
uint i ;
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
{
if ( _Tables [ i ] ! = NULL & & _Tables [ i ] - > initialised ( ) )
{
_Tables [ i ] - > releaseAll ( ) ;
}
}
return true ;
}
//
// Set method
//
bool CDatabase : : set ( RY_PDS : : TTableIndex table , RY_PDS : : TRowIndex row , RY_PDS : : TColumnIndex column , uint datasize , const void * dataptr )
{
H_AUTO ( PDS_Database_set ) ;
if ( ! initialised ( ) )
{
PDS_WARNING ( " set(): database not initialised " ) ;
return false ;
}
CTable * _table = getNonConstTable ( table ) ;
if ( _table = = NULL )
{
PDS_WARNING ( " set(): table '%d' is NULL " , table ) ;
return false ;
}
bool success = _table - > set ( row , column , datasize , dataptr ) ;
if ( success )
PDS_FULL_DEBUG ( " set '%s' column '%d' successfully " , RY_PDS : : CObjectIndex ( table , row ) . toString ( ) . c_str ( ) , column ) ;
return success ;
}
/*
* Set an object parent
*/
bool CDatabase : : setParent ( RY_PDS : : TTableIndex table , RY_PDS : : TRowIndex row , RY_PDS : : TColumnIndex column , const RY_PDS : : CObjectIndex & parent )
{
H_AUTO ( PDS_Database_setParent ) ;
if ( ! initialised ( ) )
{
PDS_WARNING ( " set(): database not initialised " ) ;
return false ;
}
CTable * _table = getNonConstTable ( table ) ;
if ( _table = = NULL )
{
PDS_WARNING ( " set(): table '%d' is NULL " , table ) ;
return false ;
}
bool success = _table - > setParent ( row , column , parent ) ;
if ( success )
PDS_FULL_DEBUG ( " set '%s' column '%d' successfully " , RY_PDS : : CObjectIndex ( table , row ) . toString ( ) . c_str ( ) , column ) ;
return success ;
}
//
// Get method
//
bool CDatabase : : get ( RY_PDS : : TTableIndex table , RY_PDS : : TRowIndex row , RY_PDS : : TColumnIndex column , uint & datasize , void * dataptr , TDataType & type )
{
if ( ! initialised ( ) )
{
PDS_WARNING ( " set(): database not initialised " ) ;
return false ;
}
CTable * _table = getNonConstTable ( table ) ;
if ( _table = = NULL )
{
PDS_WARNING ( " set(): table '%d' is NULL " , table ) ;
return false ;
}
bool success = _table - > get ( row , column , datasize , dataptr , type ) ;
if ( success )
PDS_FULL_DEBUG ( " get '%s' column '%d' successfully " , RY_PDS : : CObjectIndex ( table , row ) . toString ( ) . c_str ( ) , column ) ;
return success ;
}
/*
* Display database
*/
void CDatabase : : display ( NLMISC : : CLog * log , bool displayHeader ) const
{
if ( ! initialised ( ) )
{
log - > displayNL ( " Database not initialised " ) ;
}
log - > displayNL ( " Database '%s' " , _State . Name . c_str ( ) ) ;
log - > displayNL ( " %d types, %d tables " , _Types . size ( ) , _Tables . size ( ) ) ;
uint i ;
log - > displayNL ( " %-36s | %-15s | %-2s | %s " , " TypeId/Name " , " DataTypeId/Name " , " Sz " , " IsIndex " ) ;
for ( i = 0 ; i < _Types . size ( ) ; + + i )
{
if ( _Types [ i ] = = NULL )
log - > display ( " ** Type %d not initialised " , i ) ;
else
_Types [ i ] - > display ( log ) ;
}
bool first = true ;
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
{
if ( _Tables [ i ] = = NULL )
{
log - > display ( " ** Table %d not initialised " , i ) ;
}
else
{
_Tables [ i ] - > display ( log , false , first ) ;
first = false ;
}
}
}
/*
* Dump database content and info of an object to xml
*/
void CDatabase : : dumpToXml ( const RY_PDS : : CObjectIndex & index , NLMISC : : IStream & xml , sint expandDepth )
{
if ( xml . isReading ( ) | | ! initialised ( ) )
return ;
CTable * table = getNonConstTable ( index . table ( ) ) ;
if ( table ! = NULL )
table - > dumpToXml ( index . row ( ) , xml , expandDepth ) ;
}
/*
* Set value with human readable parameters
*/
bool CDatabase : : set ( RY_PDS : : TTableIndex table , RY_PDS : : TRowIndex row , RY_PDS : : TColumnIndex column , const std : : string & type , const std : : string & value )
{
TDataType datatype = getDataTypeFromName ( type ) ;
if ( ! checkDataType ( datatype ) )
return false ;
switch ( datatype )
{
case PDS_bool :
case PDS_char :
case PDS_uint8 :
case PDS_sint8 :
{
2010-10-23 13:06:17 +00:00
uint8 data ;
NLMISC : : fromString ( value , data ) ;
2010-05-06 00:08:41 +00:00
return set ( table , row , column , sizeof ( data ) , & data ) ;
}
break ;
case PDS_ucchar :
case PDS_uint16 :
case PDS_sint16 :
{
2010-10-23 13:06:17 +00:00
uint16 data ;
NLMISC : : fromString ( value , data ) ;
2010-05-06 00:08:41 +00:00
return set ( table , row , column , sizeof ( data ) , & data ) ;
}
break ;
case PDS_uint32 :
case PDS_sint32 :
case PDS_enum :
case PDS_CSheetId :
case PDS_CNodeId :
{
2010-10-23 13:06:17 +00:00
uint32 data ;
NLMISC : : fromString ( value , data ) ;
2010-05-06 00:08:41 +00:00
return set ( table , row , column , sizeof ( data ) , & data ) ;
}
break ;
case PDS_uint64 :
case PDS_sint64 :
{
uint64 data ;
sscanf ( value . c_str ( ) , " %016 " NL_I64 " d " , & data ) ;
return set ( table , row , column , sizeof ( data ) , & data ) ;
}
break ;
case PDS_CEntityId :
{
CEntityId data ;
data . fromString ( value . c_str ( ) ) ;
return set ( table , row , column , sizeof ( data ) , & data ) ;
}
break ;
case PDS_float :
{
float data = ( float ) atof ( value . c_str ( ) ) ;
return set ( table , row , column , sizeof ( data ) , & data ) ;
}
break ;
case PDS_double :
{
double data = ( double ) atof ( value . c_str ( ) ) ;
return set ( table , row , column , sizeof ( data ) , & data ) ;
}
break ;
case PDS_Index :
{
RY_PDS : : CObjectIndex data ;
data . fromString ( value . c_str ( ) ) ;
return set ( table , row , column , sizeof ( data ) , & data ) ;
}
break ;
case PDS_dimension :
{
2010-10-23 13:06:17 +00:00
uint32 data ;
NLMISC : : fromString ( value , data ) ;
2010-05-06 00:08:41 +00:00
return set ( table , row , column , sizeof ( data ) , & data ) ;
}
break ;
}
return false ;
}
/*
* Fetch data
*/
bool CDatabase : : fetch ( const RY_PDS : : CObjectIndex & index , RY_PDS : : CPData & data , bool fetchIndex )
{
H_AUTO ( PDS_Database_fetch ) ;
if ( ! initialised ( ) )
{
PDS_WARNING ( " set(): database not initialised " ) ;
return false ;
}
CTable * table = getNonConstTable ( index . table ( ) ) ;
if ( table = = NULL )
{
PDS_WARNING ( " fetch(): unable to get table '%d' " , index . table ( ) ) ;
return false ;
}
bool success = table - > fetch ( index . row ( ) , data , fetchIndex ) ;
if ( success )
PDS_FULL_DEBUG ( " fetch '%s' successfully " , index . toString ( ) . c_str ( ) ) ;
return success ;
}
/*
* Build index allocators
* One per table
*/
bool CDatabase : : buildIndexAllocators ( std : : vector < RY_PDS : : CIndexAllocator > & allocators )
{
PDS_DEBUG ( " buildIndexAllocators() " ) ;
if ( ! initialised ( ) )
{
PDS_WARNING ( " buildIndexAllocators(): database not initialised " ) ;
return false ;
}
allocators . clear ( ) ;
allocators . resize ( _Tables . size ( ) ) ;
uint i ;
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
if ( _Tables [ i ] ! = NULL & & _Tables [ i ] - > initialised ( ) )
_Tables [ i ] - > buildIndexAllocator ( allocators [ i ] ) ;
return true ;
}
/*
* Rebuild forwardrefs from backrefs
*/
bool CDatabase : : rebuildForwardRefs ( )
{
if ( ! initialised ( ) )
{
PDS_WARNING ( " rebuildForwardRefs(): database not initialised " ) ;
return false ;
}
// clear up object list
_SetMap . clear ( ) ;
// the rebuild forwardrefs
uint i ;
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
{
if ( _Tables [ i ] ! = NULL & & _Tables [ i ] - > initialised ( ) )
{
if ( ! _Tables [ i ] - > resetForwardRefs ( ) )
{
PDS_WARNING ( " rebuildForwardRefs(): failed to resetForwardRefs(), abort " ) ;
return false ;
}
}
}
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
{
if ( _Tables [ i ] ! = NULL & & _Tables [ i ] - > initialised ( ) )
{
if ( ! _Tables [ i ] - > rebuildForwardRefs ( ) )
{
PDS_WARNING ( " rebuildForwardRefs(): failed to rebuildForwardRefs(), abort " ) ;
return false ;
}
}
}
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
{
if ( _Tables [ i ] ! = NULL & & _Tables [ i ] - > initialised ( ) )
{
if ( ! _Tables [ i ] - > fixForwardRefs ( ) )
{
PDS_WARNING ( " rebuildForwardRefs(): failed to fixForwardRefs(), abort " ) ;
return false ;
}
}
}
return true ;
}
/*
* Rebuild table maps
*/
bool CDatabase : : rebuildTableMaps ( )
{
if ( ! initialised ( ) )
{
PDS_WARNING ( " rebuildTableMaps(): database not initialised " ) ;
return false ;
}
uint i ;
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
{
if ( _Tables [ i ] ! = NULL & & _Tables [ i ] - > initialised ( ) )
{
_Tables [ i ] - > buildRowMapper ( ) ;
}
}
return true ;
}
/*
* Reset dirty lists
*/
bool CDatabase : : resetDirtyTags ( )
{
if ( ! initialised ( ) )
{
PDS_WARNING ( " resetDirtyLists(): database not initialised " ) ;
return false ;
}
uint i ;
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
{
if ( _Tables [ i ] ! = NULL & & _Tables [ i ] - > initialised ( ) )
{
_Tables [ i ] - > resetDirtyTags ( ) ;
}
}
return true ;
}
/*
* Reset and rebuild all maps , references , lists . . .
*/
bool CDatabase : : rebuildVolatileData ( )
{
PDS_DEBUG ( " rebuildVolatileData() " ) ;
if ( ! initialised ( ) )
{
PDS_WARNING ( " rebuildVolatileData(): database not initialised " ) ;
return false ;
}
// force database to notify a new reference is up to date
notifyNewReference ( ) ;
if ( ! rebuildTableMaps ( ) )
{
PDS_WARNING ( " rebuildVolatileData(): failed to rebuildTableMaps() " ) ;
return false ;
}
if ( ! rebuildForwardRefs ( ) )
{
PDS_WARNING ( " rebuildVolatileData(): failed to rebuildForwardRefs() " ) ;
return false ;
}
/*
if ( ! resetDirtyTags ( ) )
{
PDS_WARNING ( " rebuildVolatileData(): failed to resetDirtyTags() " ) ;
return false ;
}
*/
// preload data from reference files
uint i ;
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
{
if ( _Tables [ i ] ! = NULL & & _Tables [ i ] - > initialised ( ) )
{
if ( ! _Tables [ i ] - > preloadRefFiles ( ) )
{
PDS_WARNING ( " rebuildVolatileData(): failed to preloadRefFiles() for table '%d' " , i ) ;
return false ;
}
}
}
// load string manager
// if (!_StringManager.load(_Reference.getPath()))
// {
// PDS_WARNING("rebuildVolatileData(): failed to load string manager");
// }
return true ;
}
/*
* Adapt database to new description
* \ param description is the latest xml description of the database
* \ returns true is adaptation succeded
*/
CDatabase * CDatabase : : adapt ( const string & description )
{
PDS_DEBUG ( " adapt() " ) ;
if ( ! initialised ( ) )
{
PDS_WARNING ( " adapt(): failed, database not initialised " ) ;
return NULL ;
}
// get 'From' HashKey
CHashKey hash1 = _Description . getHashKey ( ) ;
// get 'Into' HashKey
2010-05-18 14:53:47 +00:00
CHashKey hash2 = getSHA1 ( ( const uint8 * ) ( description . c_str ( ) ) , ( uint32 ) description . size ( ) ) ;
2010-05-06 00:08:41 +00:00
// same hash, ok go on
if ( hash1 = = hash2 )
return this ;
// build a clean reference if needed
if ( ! isReferenceUpToDate ( ) )
{
if ( ! buildReference ( ) )
{
PDS_WARNING ( " adapt(): failed to buildReference() " ) ;
return false ;
}
}
// backup old reference
if ( ! _Reference . save ( _Reference . getPath ( ) + " ref " ) )
{
PDS_WARNING ( " adapt(): failed to backup reference index " ) ;
}
// create a new destination database
CDatabase * into = new CDatabase ( _State . Id ) ;
if ( ! into - > createFromScratch ( description ) )
{
PDS_WARNING ( " adapt(): failed to create new reference " ) ;
delete into ;
return NULL ;
}
CDatabaseAdapter adapter ;
if ( ! adapter . build ( this , into ) )
{
PDS_WARNING ( " adapt(): failed to build() adapter " ) ;
delete into ;
return NULL ;
}
if ( ! adapter . buildContent ( ) )
{
PDS_WARNING ( " adapt(): adapter failed to buildContent() " ) ;
delete into ;
return NULL ;
}
// rebuild volatile data
// that is all non persistant data that can be found from persistant data
if ( ! into - > rebuildVolatileData ( ) )
{
PDS_WARNING ( " adapt(): failed to rebuildVolatileData() " ) ;
delete into ;
return NULL ;
}
// init timestamps as database is ready to run
into - > initTimestamps ( ) ;
return into ;
}
/*
* Load previous database state
*/
bool CDatabase : : loadState ( )
{
PDS_DEBUG ( " loadState() " ) ;
if ( initialised ( ) )
{
PDS_WARNING ( " loadState(): cannot loadState(), database is already initialised " ) ;
return false ;
}
// check directory exists
std : : string directory = _Reference . getNominalRootPath ( ) ;
if ( ! CFile : : isExists ( directory ) | | ! CFile : : isDirectory ( directory ) )
{
return false ;
}
// load database current state (or previous state if current state corrupted)
if ( ! _State . load ( _Reference ) & & ! _State . load ( _Reference , true ) )
{
return false ;
}
// get last valid reference
if ( ! _Reference . load ( ) )
{
return false ;
}
if ( _Reference . Index ! = _State . CurrentIndex )
{
PDS_WARNING ( " loadState(): failed, Reference and State files have different current index " , ( _Reference . getPath ( ) + " description.xml " ) . c_str ( ) ) ;
return false ;
}
// load description
if ( ! _Description . loadDescriptionFile ( _Reference . getPath ( ) + " description.xml " ) )
{
PDS_WARNING ( " loadState(): failed to load description file '%s' " , ( _Reference . getPath ( ) + " description.xml " ) . c_str ( ) ) ;
return false ;
}
// and init!
if ( ! init ( ) )
{
PDS_WARNING ( " loadState(): failed to init db '%d' " , _State . Id ) ;
return false ;
}
PDS_DEBUG ( " loadState(): database '%d' init ok " , _State . Id ) ;
// check reference is up to date
if ( ! isReferenceUpToDate ( ) )
{
if ( ! buildReference ( ) )
{
PDS_WARNING ( " loadState(): failed to buildReference() " ) ;
return false ;
}
}
// rebuild volatile data
// that is all non persistant data that can be found from persistant data
if ( ! rebuildVolatileData ( ) )
{
PDS_WARNING ( " loadState(): failed to rebuildVolatileData() " ) ;
return false ;
}
// init timestamps as database is ready to run
initTimestamps ( ) ;
return true ;
}
/*
* Check if reference is still the same
* Reinit reference if reference changed
*/
bool CDatabase : : checkReferenceChange ( )
{
CRefIndex ref ( _State . Id ) ;
// load current reference
if ( ! ref . load ( ) )
{
PDS_WARNING ( " checkReferenceChange(): failed to load reference for database '%d', no change assumed " , _State . Id ) ;
return false ;
}
// check current database reference and current up to date reference
if ( ref . Index ! = _Reference . Index )
{
// check nothing went wrong
if ( ref . Index < _Reference . Index | | ref . Timestamp < = _Reference . Timestamp )
{
PDS_WARNING ( " checkReferenceChange(): index changed (new index %d), but is lower than previous index (%d), no change assumed " , ref . Index , _Reference . Index ) ;
return false ;
}
// affect new reference
_Reference = ref ;
notifyNewReference ( false ) ;
return true ;
}
return false ;
}
/*
* Check if reference is up to date
* Returns true if reference is the latest valid database image
*/
bool CDatabase : : isReferenceUpToDate ( )
{
PDS_DEBUG ( " isReferenceUpToDate() " ) ;
if ( ! initialised ( ) )
{
PDS_WARNING ( " isReferenceUpToDate(): failed, database is not initialised " ) ;
return false ;
}
vector < string > files ;
NLMISC : : CPath : : getPathContent ( _Reference . getRootPath ( ) + " hours " , false , false , true , files ) ;
NLMISC : : CPath : : getPathContent ( _Reference . getRootPath ( ) + " minutes " , false , false , true , files ) ;
NLMISC : : CPath : : getPathContent ( _Reference . getRootPath ( ) + " seconds " , false , false , true , files ) ;
bool upToDate = true ;
uint i ;
for ( i = 0 ; i < files . size ( ) ; + + i )
{
uint32 tableId ;
CTimestamp timestamp ;
bool deltaFile = CDBDeltaFile : : isDeltaFileName ( files [ i ] , tableId , timestamp ) ;
// bool stringLogFile = RY_PDS::CPDStringManager::isLogFileName(files[i], timestamp);
if ( deltaFile /*|| stringLogFile*/ )
{
if ( timestamp > _State . EndTimestamp )
{
CFile : : moveFile ( ( files [ i ] + " .disabled " ) . c_str ( ) , files [ i ] . c_str ( ) ) ;
continue ;
}
// compare file timestamp (in name) to current reference timestamp
if ( _Reference . Timestamp < = timestamp & & timestamp < = _State . EndTimestamp )
upToDate = false ;
}
}
if ( upToDate )
PDS_DEBUG ( " isReferenceUpToDate(): database is up to date " ) ;
return upToDate ;
}
/*
* Build a up to date reference
*/
bool CDatabase : : buildReference ( )
{
PDS_DEBUG ( " buildReference() " ) ;
CRefIndex prev = _Reference ;
CRefIndex next ( _State . Id ) ;
if ( ! next . buildNext ( ) )
{
PDS_WARNING ( " buildReference(): failed to build next ref " ) ;
return false ;
}
PDS_DEBUG ( " buildReference(): building from reference '%08X' to reference '%08X' " , prev . Index , next . Index ) ;
// build reference but clamp to last valid state (
next . Timestamp = _State . EndTimestamp ;
_Reference = next ;
// setup reference
// this will release all references owned by the internal table buffers
// so we can copy/modify reference files without problem
if ( ! CReferenceBuilder : : build ( prev , next ) )
{
PDS_WARNING ( " buildReference(): failed to build next reference " ) ;
return false ;
}
notifyNewReference ( ) ;
PDS_DEBUG ( " buildReference(): new reference is up to date " ) ;
return true ;
}
/*
* Create new database from scratch , setup everything needed ( references , etc . )
* \ param description is the xml database description
*/
bool CDatabase : : createFromScratch ( const string & description )
{
PDS_DEBUG ( " createFromScratch() " ) ;
// In order to create a new database from scratch, directory must no exist
if ( CFile : : isExists ( _Reference . getNominalRootPath ( ) ) )
{
PDS_WARNING ( " createFromScratch(): failed, directory '%s' already exists for database '%d', manual cleanup is needed to proceed " , _Reference . getRootPath ( ) . c_str ( ) , _Reference . DatabaseId ) ;
return false ;
}
// build a brand new reference
if ( ! _Reference . buildNext ( ) )
{
PDS_WARNING ( " createFromScratch(): failed, unable to init reference " ) ;
return false ;
}
if ( initialised ( ) )
{
PDS_WARNING ( " createFromScratch(): failed, database is initialised " ) ;
return false ;
}
if ( ! _Description . loadDescription ( ( uint8 * ) description . c_str ( ) ) )
{
PDS_WARNING ( " createFromScratch(): failed to load description " ) ;
return false ;
}
if ( ! init ( ) )
{
PDS_WARNING ( " createFromScratch(): failed to init() database " ) ;
return false ;
}
// // save description
// if (!_Description.saveDescription(_Reference.getPath() + "description.xml"))
// {
// PDS_WARNING("createFromScratch(): failed to save description");
// return false;
// }
// init timestamps as database is ready to run
initTimestamps ( ) ;
// a new reference is valid
notifyNewReference ( ) ;
return true ;
}
/*
* Build the delta files and purge all dirty rows in tables
*/
bool CDatabase : : buildDelta ( const CTimestamp & starttime , const CTimestamp & endtime )
{
H_AUTO ( PDS_Database_buildDelta ) ;
if ( ! initialised ( ) )
{
PDS_WARNING ( " buildDelta(): failed, database is not initialised " ) ;
return false ;
}
uint i ;
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
{
if ( ! _Tables [ i ] - > buildDelta ( starttime , endtime ) )
{
PDS_WARNING ( " buildDelta(): failed to buildDelta() for table '%d' '%s' " , i , _Tables [ i ] - > getName ( ) . c_str ( ) ) ;
return false ;
}
}
// std::string logDir = RY_PDS::CPDSLib::getLogDirectory(_State.Id);
// if (!CFile::isExists(logDir) || !CFile::isDirectory(logDir))
// {
// if (!CFile::createDirectoryTree(logDir))
// {
// PDS_WARNING("buildDelta(): failed to create log root directory '%s'", logDir.c_str());
// }
//
// if (!CFile::setRWAccess(logDir))
// {
// PDS_WARNING("buildDelta(): failed, can't set RW access to directory '%s'", logDir.c_str());
// }
// }
// // save string manager logs
// if (!_StringManager.logEmpty())
// {
// COFile smf;
// COXml smxml;
// std::string smfilename = logDir + endtime.toString()+".string_log";
// if (!smf.open(smfilename) || !smxml.init(&smf) || !_StringManager.storeLog(smxml))
// {
// PDS_WARNING("buildDelta(): failed to build string manager log file '%s'", smfilename.c_str());
// }
// }
// save straight logs
if ( ! _LogQueue . empty ( ) )
{
// std::string logfilename = logDir + endtime.toString()+"_0000.pd_log";
// COFile logf;
// if (logf.open(logfilename))
// {
// try
// {
// logf.serialCont(_LogQueue);
// }
2011-06-03 08:35:25 +00:00
// catch (const Exception& e)
2010-05-06 00:08:41 +00:00
// {
// PDS_WARNING("buildDelta(): exception occured while saving straight log : %s", e.what());
// }
// }
// else
// {
// PDS_WARNING("buildDelta(): failed to build log file '%s'", logfilename.c_str());
// }
//
_LogQueue . clear ( ) ;
}
// State file swapping
std : : string statePath = _Reference . getRootPath ( ) ;
std : : string stateName = CDatabaseState : : fileName ( ) ;
if ( CFile : : fileExists ( statePath + stateName ) & &
2010-07-01 09:51:27 +00:00
! CFile : : copyFile ( statePath + " previous_ " + stateName , statePath + stateName , false ) )
2010-05-06 00:08:41 +00:00
{
PDS_WARNING ( " buildDelta(): failed copy state file to backup previous_state " ) ;
}
// setup state timestamp
_State . EndTimestamp = endtime ;
_State . save ( _Reference ) ;
return true ;
}
/*
* Flush database from released rows
*/
bool CDatabase : : flushReleased ( )
{
if ( ! initialised ( ) )
{
PDS_WARNING ( " flushReleased(): failed, database is not initialised " ) ;
return false ;
}
uint i ;
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
{
if ( ! _Tables [ i ] - > flushReleased ( ) )
{
PDS_WARNING ( " flushReleased(): failed to flushReleased() for table '%d' '%s' " , i , _Tables [ i ] - > getName ( ) . c_str ( ) ) ;
return false ;
}
}
return true ;
}
/*
* Notify a new reference is ready
*/
bool CDatabase : : notifyNewReference ( bool validateRef )
{
if ( ! initialised ( ) )
{
PDS_WARNING ( " notifyNewReference(): failed, database is not initialised " ) ;
return false ;
}
if ( validateRef )
_Reference . setAsValidRef ( ) ;
_State . CurrentIndex = _Reference . Index ;
uint i ;
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
_Tables [ i ] - > notifyNewReference ( _Reference ) ;
return true ;
}
/*
* Receive update
*/
void CDatabase : : receiveUpdate ( uint32 id )
{
// add to acknowledged updates
_ReceivedUpdates . push_back ( id ) ;
_State . LastUpdateId = id ;
}
/*
* Flush updates
*/
void CDatabase : : flushUpdates ( std : : vector < uint32 > & acknowledged )
{
// copy acknowledge and flush
acknowledged = _ReceivedUpdates ;
_ReceivedUpdates . clear ( ) ;
}
/**
* Get Update Queue for id
* May return NULL if message was already received
*/
RY_PDS : : CDbMessageQueue * CDatabase : : getUpdateMessageQueue ( uint32 updateId )
{
if ( updateId ! = 0 & & updateId < = _State . LastUpdateId )
return NULL ;
_LogQueue . push_back ( RY_PDS : : CUpdateLog ( ) ) ;
RY_PDS : : CUpdateLog & ulog = _LogQueue . back ( ) ;
ulog . UpdateId = updateId ;
ulog . createUpdates ( ) ;
ulog . StartStamp . setToCurrent ( ) ;
ulog . EndStamp . setToCurrent ( ) ;
return ulog . getUpdates ( ) ;
}
CVariable < uint > MinuteUpdateRate ( " pds " , " MinuteUpdateRate " , " Number of seconds between two minute updates " , 60 , 0 , true ) ;
CVariable < uint > HourUpdateRate ( " pds " , " HourUpdateRate " , " Number of seconds between two hour updates " , 3600 , 0 , true ) ;
CVariable < uint > ReferenceUpdateRate ( " pds " , " ReferenceUpdateRate " , " Number of seconds between two reference builds " , 86400 , 0 , true ) ;
CVariable < uint > KeepSecondsAtMinuteUpdate ( " pds " , " KeepSecondsAtMinuteUpdate " , " Number of seconds to keep (in number of minute updates) " , 2 , 0 , true ) ;
CVariable < uint > KeepMinutesAtHourUpdate ( " pds " , " KeepMinutesAtHourUpdate " , " Number of minutes to keep (in number of hours updates) " , 2 , 0 , true ) ;
CVariable < uint > KeepHoursAtReferenceUpdate ( " pds " , " KeepHoursAtReferenceUpdate " , " Number of hours to keep (in number of daily updates) " , 2 , 0 , true ) ;
/*
* Send Delta / Reference build commands
*/
bool CDatabase : : sendBuildCommands ( const CTimestamp & current )
{
checkUpdateRates ( ) ;
while ( current - _MinuteUpdateTimestamp > = MinuteUpdateRate )
{
CTimestamp start = _MinuteUpdateTimestamp ;
CTimestamp end = start + MinuteUpdateRate ;
CTimestamp keep = end - MinuteUpdateRate * KeepSecondsAtMinuteUpdate ;
string outputPath = _Reference . getMinutesUpdatePath ( ) ;
string hoursUpdatePath = _Reference . getHoursUpdatePath ( ) ;
string minutesUpdatePath = _Reference . getMinutesUpdatePath ( ) ;
string secondsUpdatePath = _Reference . getSecondsUpdatePath ( ) ;
string mintimestamp = start . toString ( ) ;
string maxtimestamp = end . toString ( ) ;
CDeltaBuilder : : TDelta type = CDeltaBuilder : : Minute ;
string keeptimestamp = keep . toString ( ) ;
CMessage & msgdelta = CDbManager : : addTask ( " RB_GEN_DELTA " , NULL , NULL ) ;
msgdelta . serial ( outputPath ) ;
msgdelta . serial ( hoursUpdatePath ) ;
msgdelta . serial ( minutesUpdatePath ) ;
msgdelta . serial ( secondsUpdatePath ) ;
msgdelta . serial ( mintimestamp ) ;
msgdelta . serial ( maxtimestamp ) ;
msgdelta . serialEnum ( type ) ;
msgdelta . serial ( keeptimestamp ) ;
_MinuteUpdateTimestamp = end ;
}
while ( current - _HourUpdateTimestamp > = HourUpdateRate )
{
CTimestamp start = _HourUpdateTimestamp ;
CTimestamp end = start + HourUpdateRate ;
CTimestamp keep = end - HourUpdateRate * KeepMinutesAtHourUpdate ;
string outputPath = _Reference . getHoursUpdatePath ( ) ;
string hoursUpdatePath = _Reference . getHoursUpdatePath ( ) ;
string minutesUpdatePath = _Reference . getMinutesUpdatePath ( ) ;
string secondsUpdatePath = _Reference . getSecondsUpdatePath ( ) ;
string mintimestamp = start . toString ( ) ;
string maxtimestamp = end . toString ( ) ;
CDeltaBuilder : : TDelta type = CDeltaBuilder : : Hour ;
string keeptimestamp = keep . toString ( ) ;
CMessage & msgdelta = CDbManager : : addTask ( " RB_GEN_DELTA " , NULL , NULL ) ;
msgdelta . serial ( outputPath ) ;
msgdelta . serial ( hoursUpdatePath ) ;
msgdelta . serial ( minutesUpdatePath ) ;
msgdelta . serial ( secondsUpdatePath ) ;
msgdelta . serial ( mintimestamp ) ;
msgdelta . serial ( maxtimestamp ) ;
msgdelta . serialEnum ( type ) ;
msgdelta . serial ( keeptimestamp ) ;
_HourUpdateTimestamp = end ;
}
while ( current - _DayUpdateTimestamp > = ReferenceUpdateRate )
{
CTimestamp start = _DayUpdateTimestamp ;
CTimestamp end = start + ReferenceUpdateRate ;
CTimestamp keep = end - ReferenceUpdateRate * KeepHoursAtReferenceUpdate ;
CRefIndex * next = new CRefIndex ( _State . Id ) ;
* next = _Reference ;
next - > buildNext ( ) ;
string rootRefPath = _Reference . getRootPath ( ) ;
string previousReferencePath = _Reference . getPath ( ) ;
string nextReferencePath = next - > getPath ( ) ;
string logUpdatePath = _Reference . getLogPath ( ) ;
string hoursUpdatePath = _Reference . getHoursUpdatePath ( ) ;
string minutesUpdatePath = _Reference . getMinutesUpdatePath ( ) ;
string secondsUpdatePath = _Reference . getSecondsUpdatePath ( ) ;
string mintimestamp = start . toString ( ) ;
string maxtimestamp = end . toString ( ) ;
string keeptimestamp = keep . toString ( ) ;
CMessage & msgref = CDbManager : : addTask ( " RB_GEN_REF " , this , ( void * ) next ) ;
msgref . serial ( rootRefPath ) ;
msgref . serial ( previousReferencePath ) ;
msgref . serial ( nextReferencePath ) ;
msgref . serial ( hoursUpdatePath ) ;
msgref . serial ( minutesUpdatePath ) ;
msgref . serial ( secondsUpdatePath ) ;
msgref . serial ( logUpdatePath ) ;
msgref . serial ( mintimestamp ) ;
msgref . serial ( maxtimestamp ) ;
msgref . serial ( keeptimestamp ) ;
_DayUpdateTimestamp = end ;
}
return true ;
}
/*
* Task ran successfully
*/
void CDatabase : : taskSuccessful ( void * arg )
{
// when reference is up to date, notify new reference
CRefIndex * ref = ( CRefIndex * ) arg ;
_Reference = * ref ;
notifyNewReference ( true ) ;
delete ref ;
}
/*
* Task failed !
*/
void CDatabase : : taskFailed ( void * arg )
{
// ok, nothing to do in this case...
}
/*
* Static Check for update rates
*/
void CDatabase : : checkUpdateRates ( )
{
uint deltaRate = DeltaUpdateRate ;
uint minuteRate = MinuteUpdateRate ;
// minute rate must be a multiple of delta rate
if ( ( minuteRate % deltaRate ) ! = 0 )
{
minuteRate = deltaRate * ( minuteRate / deltaRate + 1 ) ;
nlwarning ( " CDatabase::checkUpdateRate(): MinuteUpdateRate is not a multiple of DeltaUpdateRate! Rounded to %d seconds " , minuteRate ) ;
MinuteUpdateRate = minuteRate ;
}
uint hourRate = HourUpdateRate ;
// hour rate must be a multiple of minute rate
if ( ( hourRate % minuteRate ) ! = 0 )
{
hourRate = minuteRate * ( hourRate / minuteRate + 1 ) ;
nlwarning ( " CDatabase::checkUpdateRate(): HourUpdateRate is not a multiple of MinuteUpdateRate! Rounded to %d seconds " , hourRate ) ;
HourUpdateRate = hourRate ;
}
uint referenceRate = ReferenceUpdateRate ;
// reference rate must be a multiple of hour rate
if ( ( referenceRate % hourRate ) ! = 0 )
{
referenceRate = hourRate * ( referenceRate / hourRate + 1 ) ;
nlwarning ( " CDatabase::checkUpdateRate(): ReferenceUpdateRate is not a multiple of HourUpdateRate! Rounded to %d seconds " , referenceRate ) ;
ReferenceUpdateRate = referenceRate ;
}
}
/*
* Serialise SheetId String Mapper
*/
void CDatabase : : serialSheetIdStringMapper ( NLMISC : : IStream & f )
{
// serial mapper
_SheetIdStringMapper . serial ( f ) ;
if ( f . isReading ( ) )
{
// if mapper is read, save it now in reference path
std : : string refPath = _Reference . getPath ( ) ;
COFile ofile ;
if ( ! ofile . open ( refPath + " sheetid_map.bin " ) )
{
PDS_WARNING ( " serialSheetIdStringMapper(): failed to open reference sheetid_map file '%s' for save " , ( refPath + " sheetid_map.bin " ) . c_str ( ) ) ;
return ;
}
_SheetIdStringMapper . serial ( ofile ) ;
}
}
/*
* Get Table Index from name
*/
RY_PDS : : TTableIndex CDatabase : : getTableIndex ( const std : : string & tableName ) const
{
const CTable * table = getTable ( tableName ) ;
return ( table = = NULL ) ? RY_PDS : : INVALID_TABLE_INDEX : ( RY_PDS : : TTableIndex ) table - > getId ( ) ;
}
/*
* Get Table Index from name
*/
std : : string CDatabase : : getTableName ( RY_PDS : : TTableIndex index ) const
{
if ( index > = _Tables . size ( ) | | _Tables [ index ] = = NULL )
return " invalid " ;
return _Tables [ index ] - > getName ( ) ;
}
/*
* Search object in database using its key
* \ param key is the 64 bits row key to search through all tables
*/
bool CDatabase : : searchObjectIndex ( uint64 key , std : : set < RY_PDS : : CObjectIndex > & indexes ) const
{
indexes . clear ( ) ;
uint i ;
for ( i = 0 ; i < _Tables . size ( ) ; + + i )
{
if ( _Tables [ i ] = = NULL | | ! _Tables [ i ] - > initialised ( ) | | ! _Tables [ i ] - > isMapped ( ) )
continue ;
RY_PDS : : CObjectIndex index = _Tables [ i ] - > getMappedRow ( key ) ;
if ( index . isValid ( ) )
indexes . insert ( index ) ;
}
return ! indexes . empty ( ) ;
}