khanat-opennel-code/code/ryzom/server/src/persistant_data_service/pds_table.cpp

2319 lines
50 KiB
C++
Raw Normal View History

// 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 <nel/misc/debug.h>
#include <nel/misc/common.h>
#include <nel/misc/entity_id.h>
#include <string.h>
#include "pds_table.h"
#include "pds_database.h"
//#include "db_file_stream.h"
#include "../pd_lib/db_delta_file.h"
#include "../pd_lib/db_description_parser.h"
using namespace std;
using namespace NLMISC;
/// Aligne a value on a boundary (if boundary is not a power of 2, rounded to next power of 2 boundary)
inline uint32 alignOnBoundary(uint32 value, uint32 boundary)
{
// round to power of 2 boundary
boundary = raiseToNextPowerOf2(boundary);
// align
return (value+boundary-1)&(~(boundary-1));
}
/*
* Constructor
*/
CTable::CTable()
{
clear();
}
/*
* Destructor
*/
CTable::~CTable()
{
//PDS_DEBUG("delete()");
clear();
}
/*
* Clear table
*/
void CTable::clear()
{
_Init = false;
uint i;
// clear table buffer
_TableBuffer.clear();
// delete columns
_Columns.clear();
// delete attributes
for (i=0; i<_Attributes.size(); ++i)
{
delete _Attributes[i];
_Attributes[i] = NULL;
}
_Attributes.clear();
_Parent = NULL;
_Name = "";
_Id = INVALID_TYPE_ID;
_Key = INVALID_TYPE_ID;
_Mapped = false;
_RowSize = 0;
_EmptyRow.clear();
}
/*
* Init table
*/
bool CTable::init(CDatabase *database, const CTableNode& table)
{
// first a good clean up
clear();
// set parent logger
setParentLogger(database);
_TableBuffer.setParentLogger(this);
_Parent = database;
_Name = table.Name;
_Id = table.Id;
_Inheritance = table.Inherit;
_Mapped = (table.Mapped != -1);
_Key = table.Key;
uint i;
for (i=0; i<table.Attributes.size(); ++i)
{
CAttribute* attribute = new CAttribute();
if (!attribute->init(database, this, table.Attributes[i]) || attribute->getId() != i)
return false;
_Attributes.push_back(attribute);
}
return true;
}
/*
* Build columns
*/
bool CTable::buildColumns()
{
// do not compute twice
if (!_Columns.empty())
return true;
uint i;
// build columns from attributes
for (i=0; i<_Attributes.size(); ++i)
if (!_Attributes[i]->buildColumns())
return false;
uint32 columnStart = 0;
for (i=0; i<_Columns.size(); ++i)
{
_Columns[i].setByteOffset(columnStart);
columnStart += _Columns[i].getByteSize();
}
_RowSize = columnStart;
_TableBuffer.init(_Id, _RowSize, _Mapped);
_EmptyRow.clear();
_EmptyRow.resize(_RowSize, 0);
for (i=0; i<_Columns.size(); ++i)
{
TDataType type = _Columns[i].getDataType();
void* data = &_EmptyRow[_Columns[i].getByteOffset()];
if (type == PDS_Index)
{
*(RY_PDS::CObjectIndex*)data = RY_PDS::CObjectIndex::null();
}
else if (type == PDS_List)
{
memset(data, 0, _Columns[i].getByteSize());
}
}
_Init = true;
return true;
}
/*
* Post Init, called once all tables have been initialised
*/
bool CTable::postInit()
{
CTable* root = getRootTable();
if (root != this)
_TableBuffer.linkRowMapper(&(root->_TableBuffer));
uint i;
for (i=0; i<_Attributes.size(); ++i)
if (!_Attributes[i]->computeBackRefKey())
return false;
return true;
}
/*
* Display table
*/
void CTable::display(NLMISC::CLog* log, bool expanded, bool displayHeader) const
{
if (!initialised())
{
log->displayNL("** Table not initialised");
return;
}
if (displayHeader)
{
log->displayNL("--------------------------------------------------------------------------------");
log->displayNL("%-10s %-3s %-32s | %-3s | %-3s | %-4s | %-8s %-8s %8s |", "", "Id", "Name", "Inh", "Att", "Cols", "Allocs", "Loaded", "LoadedSz");
}
log->displayNL("%-10s %-3d %-32s | %-3d | %-3d | %-4d | %-8d %-8d %6dkb |", "Table", _Id, _Name.c_str(), _Inheritance, _Attributes.size(), _Columns.size(), _TableBuffer.numAllocated(), _TableBuffer.getLoadedRows(), (_TableBuffer.getMemoryLoad()+1023)/1024);
uint i;
if (expanded)
{
log->displayNL("--------------------------------------------------------------------------------");
log->displayNL("%-10s %-3s %-32s | %-10s | %-4s %-4s |", "", "Id", "Name", "MetaType", "From", "NCol");
for (i=0; i<_Attributes.size(); ++i)
{
const CAttribute* attrib = _Attributes[i];
if (attrib == NULL || !attrib->initialised())
log->displayNL("** Attribute %d not initialised", i);
else
log->displayNL("%-10s %-3d %-32s | %-10s | %-4d %-4d |", "Attribute", attrib->getId(), attrib->getName().c_str(), getNameFromMetaType(attrib->getMetaType()).c_str(), attrib->getOffset(), attrib->getColumns());
}
log->displayNL("--------------------------------------------------------------------------------");
log->displayNL("%-10s %-3s %-64s | %-10s %-10s | %-10s", "", "Id", "Name", "MetaType", "DataType", "RowSz/Offs");
for (i=0; i<_Columns.size(); ++i)
{
const CColumn& column = _Columns[i];
if (!column.initialised())
log->displayNL("** Column %d not initialised", i);
else
log->displayNL("%-10s %-3d %-64s | %-10s %-10s | %1db at %-3d", "Column", column.getId(), column.getName().c_str(), getNameFromMetaType(column.getMetaType()).c_str(), getNameFromDataType(column.getDataType()).c_str(), column.getByteSize(), column.getByteOffset());
}
}
}
/*
* Display row
*/
void CTable::displayRow(RY_PDS::TRowIndex row, NLMISC::CLog* log, bool displayHeader)
{
if (!initialised())
{
log->displayNL("** Table not initialised");
return;
}
CTableBuffer::CAccessor rowaccess = _TableBuffer.getRow(row);
string flagstr = toString("%s%s%s",
(rowaccess.allocated() ? "allocated" : "free"),
(rowaccess.mapped() ? ", mapped" : ""),
(rowaccess.dirty() ? ", dirty" : ""));
log->displayNL("row %d: %d bytes, flags=[%s] (map=%016"NL_I64"X, dirtstamp=%08X)", row, _RowSize, flagstr.c_str(), (rowaccess.mapped() ? rowaccess.key() : (uint64)0), rowaccess.dirtyStamp());
if (displayHeader)
{
log->displayNL("%-10s %-3s %-64s | %-10s %-10s | %-9s | %-32s", "", "Id", "Name", "MetaType", "DataType", "Sz/Offs", "Value");
}
uint i;
for (i=0; i<_Columns.size(); ++i)
{
if (!_Columns[i].initialised())
{
log->displayNL("** Column %d not initialised", i);
continue;
}
string value;
CDataAccessor accessor(this, rowaccess, i);
const CColumn &col = _Columns[i];
if (!accessor.isValid())
value = "unaccessible value";
else
value = accessor.valueAsString();
log->displayNL("%-10s %-3d %-64s | %-10s %-10s | %1db at %-3d | %-32s",
"Column", col.getId(), col.getName().c_str(),
getNameFromMetaType(col.getMetaType()).c_str(), getNameFromDataType(col.getDataType()).c_str(),
col.getByteSize(), col.getByteOffset(),
value.c_str());
}
_TableBuffer.releaseRow(rowaccess);
}
/*
* Dump Delta file content
*/
void CTable::dumpDeltaFileContent(const std::string& filename, NLMISC::CLog* log) const
{
#ifdef DEBUG_DATA_ACCESSOR
if (!initialised())
{
PDS_WARNING("dumpDeltaFileContent(): table not initialised");
return;
}
CDBDeltaFile delta;
if (!_TableBuffer.setupDebugDeltaFile(filename, delta))
{
return;
}
uint8* data;
uint32 row;
log->displayNL("%-10s %-3s %-64s | %-10s %-10s | %-9s | %-32s", "", "Id", "Name", "MetaType", "DataType", "Sz/Offs", "Value");
while ((data = _TableBuffer.getDeltaRow(row, delta)) != NULL)
{
log->displayNL("row %d", row);
CTable* table = const_cast<CTable*>(this);
CDataAccessor accessor(table, data, 0);
uint i;
for (i=0; i<_Columns.size(); ++i)
{
if (!_Columns[i].initialised())
{
log->displayNL("** Column %d not initialised", i);
continue;
}
string value;
CDataAccessor accessor(accessor, i);
const CColumn &col = _Columns[i];
if (!accessor.isValid())
value = "unaccessible value";
else
value = accessor.valueAsString();
log->displayNL("%-10s %-3d %-64s | %-10s %-10s | %1db at %-3d | %-32s",
"Column", col.getId(), col.getName().c_str(),
CType::getNameFromMetaType(col.getMetaType()).c_str(), CType::getNameFromDataType(col.getDataType()).c_str(),
col.getByteSize(), col.getByteOffset(),
value.c_str());
}
}
#endif
}
/*
* Rebuild forwardrefs from backrefs
*/
bool CTable::rebuildForwardRefs()
{
if (!initialised())
{
PDS_WARNING("rebuildForwardRefs(): table not initialised");
return false;
}
if (!fillRefInfo())
{
PDS_WARNING("rebuildForwardRefs(): failed to fillRefInfo()");
return false;
}
if (BackRefInfo.empty() && ForwardRefInfo.empty())
return true;
if (!_TableBuffer.processRows(this))
{
PDS_WARNING("rebuildForwardRefs(): failed to processRows()");
return false;
}
return true;
}
/*
* Reset forwardrefs
*/
bool CTable::resetForwardRefs()
{
if (!initialised())
{
PDS_WARNING("resetForwardRefs(): table not initialised");
return false;
}
return true;
}
/*
* Reset table map
*/
bool CTable::resetTableMap()
{
if (!initialised())
{
PDS_WARNING("resetTableMap(): table not initialised");
return false;
}
return true;
}
/*
* Rebuild table map
*/
bool CTable::rebuildTableMap()
{
if (!initialised())
{
PDS_WARNING("rebuildTableMap(): table not initialised");
return false;
}
if (!_Mapped)
return true;
return true;
}
/*
* Allocate a row in a table
* \param row is the row to allocate
* Return true if succeded
*/
bool CTable::allocate(RY_PDS::TRowIndex row, bool acquireRow)
{
CTableBuffer::CAccessor accessor;
if (!_TableBuffer.allocate(row, accessor))
return false;
if (!accessor.allocated())
return false; // should not happen
resetRow(accessor.data());
// lock row if required
if (acquireRow)
_TableBuffer.acquireRow(accessor);
return true;
}
/*
* Deallocate a row in a table
* \param row is the row to deallocate
* Return true if succeded
*/
bool CTable::deallocate(RY_PDS::TRowIndex row)
{
return _TableBuffer.deallocate(row);
}
/*
* Map a row
* \param row is the row to allocate
* \param key is the 64 bits row key
* Return true if succeded
*/
bool CTable::mapRow(const RY_PDS::CObjectIndex &index, uint64 key)
{
if (!initialised())
{
PDS_WARNING("mapRow(): failed, table is not initialised");
return false;
}
if (!_Mapped)
{
PDS_WARNING("mapRow(): failed, table is not mapped");
return false;
}
if (!_TableBuffer.mapRow(index, key))
{
PDS_WARNING("mapRow(): failed to map '%s' to '%016"NL_I64"X'", index.toString().c_str(), key);
return false;
}
PDS_FULL_DEBUG("Mapped '%s' to key '%016"NL_I64"X'", index.toString().c_str(), key);
return true;
}
/*
* 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 CTable::unmapRow(uint64 key)
{
if (!initialised())
{
PDS_WARNING("unmapRow(): failed, table is not initialised");
return false;
}
if (!_Mapped)
{
PDS_WARNING("unmapRow(): failed, table is not mapped");
return false;
}
// get index
RY_PDS::CObjectIndex index = _TableBuffer.getMappedRow(key);
if (index.isNull())
{
PDS_WARNING("unmapRow(): failed, inherited table is not initialised");
return false;
}
if (!_TableBuffer.unmapRow(index, key))
{
PDS_WARNING("mapRow(): failed to unmap '%s' to '%016"NL_I64"X'", index.toString().c_str(), key);
return false;
}
PDS_FULL_DEBUG("Unmapped '%s' of key '%016"NL_I64"X'", index.toString().c_str(), key);
return true;
}
/*
* Get a mapped row
* \param key is the 64 bits row key
* Return a valid TRowIndex if success
*/
RY_PDS::CObjectIndex CTable::getMappedRow(uint64 key) const
{
if (!initialised())
{
PDS_WARNING("getMappedRow(): failed, table is not initialised");
return RY_PDS::CObjectIndex::null();
}
return _TableBuffer.getMappedRow(key);
}
/*
* Release a row in a table
* \param row is the row to release
* \param timestamp is the current timestamp
* Return true if succeded
*/
bool CTable::release(RY_PDS::TRowIndex row)
{
if (!initialised())
{
PDS_WARNING("release(): failed, table is not initialised");
return false;
}
return _TableBuffer.releaseRow(row);
}
/*
* Release all rows in table
*/
bool CTable::releaseAll()
{
return _TableBuffer.releaseAll();
}
/*
* Set value
*/
bool CTable::set(RY_PDS::TRowIndex row, RY_PDS::TColumnIndex column, uint datasize, const void* dataptr)
{
RY_PDS::CObjectIndex object((RY_PDS::TTableIndex)_Id, row);
// get an accessor on data
CDataAccessor accessor(_Parent, object, column);
if (!accessor.isValid())
{
PDS_WARNING("set(): failed to get accessor on '%s'", accessor.getColumnIndex().toString().c_str());
return false;
}
switch (accessor.column()->getMetaType())
{
case PDS_Type:
// for simple type, easy money
return accessor.setValue(dataptr, datasize);
break;
case PDS_BackRef:
{
// first unlink previous parent
if (!unlink(accessor, object))
{
PDS_WARNING("set(): unable to unlink previous parent at '%s'", accessor.getColumnIndex().toString().c_str());
}
// then link new parent
if (datasize != getStandardByteSize(PDS_Index))
{
PDS_WARNING("set(): unable to link new at '%s', provided bytesize is not standard", accessor.getColumnIndex().toString().c_str());
return false;
}
// get parent index
RY_PDS::CObjectIndex parent = *(RY_PDS::CObjectIndex*)dataptr;
// check checksum is valid
if (!parent.isChecksumValid())
{
PDS_WARNING("set(): unable to link new parent '%s' at '%s', parent checksum is invalid", parent.toString().c_str(), accessor.getColumnIndex().toString().c_str());
return false;
}
// if parent is invalid, then no need to link
if (!parent.isValid())
return true;
// and link!
if (!link(accessor, parent, object))
{
PDS_WARNING("set(): unable to link new parent '%s' at '%s'", parent.toString().c_str(), accessor.getColumnIndex().toString().c_str());
return false;
}
}
break;
// other types MUST not be directly set
default:
PDS_WARNING("set(): column '%d' doesn't support set on '%s'", column, getNameFromMetaType(accessor.column()->getMetaType()).c_str());
return false;
break;
}
return true;
}
/**
* Set Parent
*/
bool CTable::setParent(RY_PDS::TRowIndex row, RY_PDS::TColumnIndex column, const RY_PDS::CObjectIndex& parent)
{
RY_PDS::CObjectIndex object((RY_PDS::TTableIndex)_Id, row);
// get an accessor on data
CDataAccessor accessor(_Parent, object, column);
if (!accessor.isValid())
{
PDS_WARNING("set(): failed to get accessor on '%s'", accessor.getColumnIndex().toString().c_str());
return false;
}
switch (accessor.column()->getMetaType())
{
case PDS_BackRef:
{
// first unlink previous parent
if (!unlink(accessor, object))
{
PDS_WARNING("set(): unable to unlink previous parent at '%s'", accessor.getColumnIndex().toString().c_str());
}
// check checksum is valid
if (!parent.isChecksumValid())
{
PDS_WARNING("set(): unable to link new parent '%s' at '%s', parent checksum is invalid", parent.toString().c_str(), accessor.getColumnIndex().toString().c_str());
return false;
}
// if parent is invalid, then no need to link
if (!parent.isValid())
return true;
// and link!
if (!link(accessor, parent, object))
{
PDS_WARNING("set(): unable to link new parent '%s' at '%s'", parent.toString().c_str(), accessor.getColumnIndex().toString().c_str());
return false;
}
}
break;
// other types MUST not be directly set
default:
PDS_WARNING("setParent(): column '%d' doesn't support set on '%s'", column, getNameFromMetaType(accessor.column()->getMetaType()).c_str());
return false;
break;
}
return true;
}
/**
* link a BackRef to a parent
* This method will perfom a full linking of child and new parent
* \param backref is an accessor on the BackRef
* \param parent is an index on the parent to link
*/
bool CTable::link(CDataAccessor &backref, const RY_PDS::CObjectIndex &parent, const RY_PDS::CObjectIndex &child)
{
// check ref is valid
if (!backref.isValid())
{
PDS_WARNING("link(): failed, accessor is not valid");
return false;
}
// check really a BackRef
if (backref.attribute()->getMetaType() != PDS_BackRef || backref.column()->getMetaType() != PDS_BackRef)
{
PDS_WARNING("link(): failed, accessor is not a BackRef");
return false;
}
if (!parent.isValid())
{
PDS_WARNING("link(): failed, parent has not a valid index");
return false;
}
// link parent to child
uint32 forwardrefid = backref.attribute()->getReferencedAttribute();
CDataAccessor parentref(_Parent, parent, forwardrefid, 0);
if (!parentref.isValid())
{
PDS_WARNING("link(): failed, parent '%s' is not valid", parentref.toString().c_str());
return false;
}
// check parent and child can really be linked
// that is cross references match
if (parentref.attribute()->getReferencedAttribute() != backref.attribute()->getId() ||
backref.attribute()->getReferencedAttribute() != parentref.attribute()->getId())
{
PDS_WARNING("link(): failed, child '%s' and parent '%s' are not bound to be linked", backref.toString().c_str(), parentref.toString().c_str());
return false;
}
forwardLink(backref, parentref, child);
// set no ref index, with checksum validation
return backref.setIndex(parent);
}
/*
* link a ForwardRef to a child
* This method will only link parent to child
* \param forwardref is an accessor on the BackRef
* \param child is an index on the child to link
*/
bool CTable::forwardLink(CDataAccessor &backref, CDataAccessor &forwardref, const RY_PDS::CObjectIndex &child)
{
if (!backref.isValid() || !forwardref.isValid())
{
PDS_WARNING("forwardLink(): failed, accessor is not valid");
return false;
}
if (!child.isValid())
{
PDS_WARNING("forwardLink(): failed, child '%s' index is invalid", child.toString().c_str());
return false;
}
switch (forwardref.attribute()->getMetaType())
{
case PDS_ArrayRef:
// ArrayRef and ForwardRef set code is a bit shared as ArrayRef will only do
// a little seek to the good column and set the index just like a ForwardRef
// actually, datatypes are the same!
if (!forwardref.seek(backref))
{
PDS_WARNING("forwardLink(): unable to seek to key of '%s'", child.toString().c_str());
return false;
}
case PDS_ForwardRef:
{
RY_PDS::CObjectIndex prevChild;
// check no previous child
if (!forwardref.getIndex(prevChild))
{
PDS_WARNING("forwardLink(): unable to access previous child of '%s'", forwardref.toString().c_str());
}
else if (prevChild.isValid())
{
PDS_WARNING("forwardLink(): '%s' has a previous child '%s'", forwardref.toString().c_str(), prevChild.toString().c_str());
}
// set child
return forwardref.setIndex(child);
}
break;
case PDS_Set:
{
if (!forwardref.checkType(child))
{
PDS_WARNING("forwardLink(): failed, '%s' is not of attribute '%s' type", child.toString().c_str(), forwardref.attribute()->getName().c_str());
return false;
}
RY_PDS::CSetMap::CAccessor setaccess = forwardref.getSet();
if (!setaccess.isValid())
{
PDS_WARNING("forwardLink(): failed, parent '%s' set access is invalid", forwardref.toString().c_str());
return false;
}
setaccess.add(child);
// CHECK
if (!setaccess.belongsTo(child))
{
PDS_WARNING("forwardLink(): failed, child '%s' doesn't belong to parent '%s' though it just had been added", child.toString().c_str(), forwardref.toString().c_str());
return false;
}
// CHECK
return true;
}
break;
default:
PDS_WARNING("forwardLink(): failed, can't link '%s'", getNameFromMetaType(forwardref.attribute()->getMetaType()).c_str());
break;
}
return false;
}
/*
* Unlink a BackRef
* \param ref is an accessor on the BackRef
* \param child is a remember of the child index
*/
bool CTable::unlink(CDataAccessor &backref, const RY_PDS::CObjectIndex &child)
{
// check ref is valid
if (!backref.isValid())
{
PDS_WARNING("unlink(): failed, accessor is not valid");
return false;
}
// check really a BackRef
if (backref.attribute()->getMetaType() != PDS_BackRef || backref.column()->getMetaType() != PDS_BackRef)
{
PDS_WARNING("unlink(): failed, accessor is not a BackRef");
return false;
}
RY_PDS::CObjectIndex parent;
// unlink parent
if (!backref.getIndex(parent))
{
PDS_WARNING("unlink(): unable to get parent index");
}
else if (!parent.isChecksumValid())
{
PDS_WARNING("unlink(): parent index has invalid checksum");
}
else if (parent.isValid())
{
// only unlink if parent exists
// because link will perform unlink before, and child may have no parent yet
uint32 forwardrefid = backref.attribute()->getReferencedAttribute();
CDataAccessor parentref(_Parent, parent, forwardrefid, 0);
forwardUnlink(backref, parentref, child);
}
// set no ref index, with checksum validation
return backref.setIndex(RY_PDS::CObjectIndex::null());
}
/*
* unlink a ForwardRef to a child
* This method will only unlink parent to child
* \param forwardref is an accessor on the BackRef
* \param child is an index on the child to unlink
*/
bool CTable::forwardUnlink(CDataAccessor &backref, CDataAccessor &forwardref, const RY_PDS::CObjectIndex &child)
{
if (!forwardref.isValid())
{
PDS_WARNING("forwardUnlink(): failed, accessor is not valid");
return false;
}
if (!child.isValid())
{
PDS_WARNING("forwardUnlink(): failed, child '%s' index is invalid", child.toString().c_str());
return false;
}
switch (forwardref.attribute()->getMetaType())
{
case PDS_ArrayRef:
// ArrayRef and ForwardRef set code is a bit shared as ArrayRef will only do
// a little seek to the good column and set the index just like a ForwardRef
// actually, datatypes are the same!
if (!forwardref.seek(backref))
{
PDS_WARNING("forwardLink(): unable to seek to key of '%s'", child.toString().c_str());
return false;
}
case PDS_ForwardRef:
{
RY_PDS::CObjectIndex prevChild;
// check child is the one we want to unlink
if (!forwardref.getIndex(prevChild))
{
PDS_WARNING("forwardUnlink(): unable to access previous child of '%s'", forwardref.toString().c_str());
}
else if (!prevChild.isValid())
{
PDS_WARNING("forwardUnlink(): '%s' has no previous child '%s'", prevChild.toString().c_str(), forwardref.toString().c_str());
}
else if (prevChild != child)
{
PDS_WARNING("forwardUnlink(): failed, '%s' was not linked to '%s' but to '%s'", forwardref.toString().c_str(), child.toString().c_str(), prevChild.toString().c_str());
return false;
}
// set child
return forwardref.setIndex(RY_PDS::CObjectIndex::null());
}
break;
case PDS_Set:
{
RY_PDS::CSetMap::CAccessor setaccess = forwardref.getSet();
if (!setaccess.isValid())
{
PDS_WARNING("forwardUnlink(): failed, parent '%s' set access is invalid", forwardref.toString().c_str());
return false;
}
if (!setaccess.belongsTo(child))
{
PDS_WARNING("forwardUnlink(): failed, child '%s' doesn't belong to parent '%s'", child.toString().c_str(), forwardref.toString().c_str());
return false;
}
setaccess.erase(child);
if (setaccess.belongsTo(child))
{
PDS_WARNING("forwardUnlink(): failed, child '%s' still belongs to parent '%s' though it had been deleted", child.toString().c_str(), forwardref.toString().c_str());
return false;
}
return true;
}
break;
default:
PDS_WARNING("forwardUnlink(): failed, can't unlink '%s'", getNameFromMetaType(forwardref.attribute()->getMetaType()).c_str());
break;
}
return false;
}
/*
* Set value
*/
bool CTable::get(RY_PDS::TRowIndex row, RY_PDS::TColumnIndex column, uint& datasize, void* dataptr, TDataType& type)
{
RY_PDS::CObjectIndex object((RY_PDS::TTableIndex)_Id, row);
// get an accessor on data
CDataAccessor accessor(_Parent, object, column);
if (!accessor.isValid())
{
PDS_WARNING("get(): failed to get accessor on '%s'", accessor.getColumnIndex().toString().c_str());
return false;
}
// get datatype
type = accessor.column()->getDataType();
switch (accessor.column()->getMetaType())
{
case PDS_Type:
// for simple type, easy money
return accessor.getValue(dataptr, datasize);
break;
case PDS_BackRef:
case PDS_ForwardRef:
{
if (getStandardByteSize(type) > datasize)
{
PDS_WARNING("get(): databuffer too narrow to store column '%d' Index", column);
return false;
}
datasize = sizeof(RY_PDS::CObjectIndex);
return accessor.getIndex(*(RY_PDS::CObjectIndex*)dataptr);
}
break;
case PDS_Set:
{
PDS_WARNING("get(): not supported for sets");
return false;
}
break;
// other types MUST not be directly set
default:
PDS_WARNING("get(): column '%d' doesn't support get on '%s'", column, getNameFromMetaType(accessor.column()->getMetaType()).c_str());
return false;
break;
}
return true;
}
/*
* Perform column integrity check
*/
bool CTable::CDataAccessor::check() const
{
if (!isValid())
return false;
#ifdef DEBUG_DATA_ACCESSOR
if (_IsDebug)
return true;
#endif
switch (_Column->getMetaType())
{
case PDS_Type:
return checkAsTypeAccessor();
break;
case PDS_BackRef:
if (_Attribute->getMetaType() != PDS_BackRef)
{
PDS_WARNING_IN(_Table)("CDataAccessor::check(): column '%s' MetaType incoherent with attribute's definition", _Column->getName().c_str());
return false;
}
return checkAsRefAccessor();
break;
case PDS_ForwardRef:
if (_Attribute->getMetaType() != PDS_ForwardRef &&
_Attribute->getMetaType() != PDS_ArrayRef)
{
PDS_WARNING_IN(_Table)("CDataAccessor::check(): column '%s' MetaType incoherent with attribute's definition", _Column->getName().c_str());
return false;
}
return checkAsRefAccessor();
break;
case PDS_Set:
return checkAsSetAccessor();
break;
}
return false;
}
/*
* Check an accessor as a PDS_Type accessor
*/
bool CTable::CDataAccessor::checkAsTypeAccessor() const
{
// check attribute
if (_Attribute->getMetaType() != PDS_Type &&
_Attribute->getMetaType() != PDS_Class &&
_Attribute->getMetaType() != PDS_ArrayType &&
_Attribute->getMetaType() != PDS_ArrayClass)
{
PDS_WARNING_IN(_Table)("CDataAccessor::checkAsTypeAccessor(): column '%s' MetaType incoherent with attribute's definition", _Column->getName().c_str());
return false;
}
if (!checkStrictDataType(_Column->getDataType()))
{
PDS_WARNING_IN(_Table)("CDataAccessor::checkAsTypeAccessor(): column '%s' has not a strict valid datatype ('%s')", _Column->getName().c_str(), getNameFromDataType(_Column->getDataType()).c_str());
return false;
}
if (_Column->getDataType() == PDS_enum || _Column->getDataType() == PDS_dimension)
{
const CType* index = _Table->getParent()->getType(_Column->getTypeId());
if (index == NULL)
{
PDS_WARNING_IN(_Table)("CDataAccessor::checkAsTypeAccessor(): couldn't find column '%s' original type in database", _Column->getName().c_str());
return false;
}
if (!index->isIndex())
{
PDS_WARNING_IN(_Table)("CDataAccessor::checkAsTypeAccessor(): column '%s' original type is not an enum, whereas column says so", _Column->getName().c_str());
return false;
}
uint32 value;
if (!getAsIndexType(value))
{
PDS_WARNING_IN(_Table)("CDataAccessor::checkAsTypeAccessor(): couldn't getAsIndexValue() column '%s'", _Column->getName().c_str());
return false;
}
if (value >= index->getIndexSize())
{
PDS_WARNING_IN(_Table)("CDataAccessor::checkAsTypeAccessor(): column '%s' is out of enum '%s' range", _Column->getName().c_str(), index->getName().c_str());
return false;
}
}
return true;
}
/*
* Check an accessor as a PDS_BackRef or PDS_ForwardRef accessor
*/
bool CTable::CDataAccessor::checkAsRefAccessor() const
{
if (_Column->getDataType() != PDS_Index)
{
PDS_WARNING_IN(_Table)("CDataAccessor::checkAsRefAccessor(): column '%s' DataType must be Index, found '%s'", _Column->getName().c_str(), getNameFromDataType(_Column->getDataType()).c_str());
return false;
}
RY_PDS::CObjectIndex ref;
if (!getIndex(ref))
{
PDS_WARNING_IN(_Table)("CDataAccessor::checkAsRefAccessor(): can't get column '%s' Index", _Column->getName().c_str());
return false;
}
if (!checkType(ref))
{
PDS_WARNING_IN(_Table)("CDataAccessor::checkAsRefAccessor(): column '%s' contains object '%s' of invalid type", _Column->getName().c_str(), ref.toString().c_str());
return false;
}
if (!ref.isNull() && !_Table->getParent()->isAllocated(ref))
{
PDS_WARNING_IN(_Table)("CDataAccessor::checkAsRefAccessor(): column '%s' points to '%s' which is not allocated", _Column->getName().c_str(), ref.toString().c_str());
return false;
}
return true;
}
/*
* Check an accessor as a PDS_Set accessor
*/
bool CTable::CDataAccessor::checkAsSetAccessor() const
{
if (_Column->getDataType() != PDS_List)
{
PDS_WARNING_IN(_Table)("CDataAccessor::checkAsSetAccessor(): column '%s' DataType must be List, found '%s'", _Column->getName().c_str(), getNameFromDataType(_Column->getDataType()).c_str());
return false;
}
RY_PDS::CSetMap::CAccessor setaccess = getSet();
if (!setaccess.isValid())
{
PDS_WARNING_IN(_Table)("CDataAccessor::checkAsSetAccessor(): set accessor to '%s' is invalid", toString().c_str());
return false;
}
const RY_PDS::TIndexList& list = setaccess.get();
bool success = true;
uint i;
for (i=0; i<list.size(); ++i)
{
RY_PDS::CObjectIndex index = list[i];
if (!index.isValid())
{
PDS_WARNING_IN(_Table)("CDataAccessor::checkAsSetAccessor(): column '%s' points to invalid object '%s'", _Column->getName().c_str(), index.toString().c_str());
success = false;
continue;
}
if (!_Table->getParent()->isAllocated(index))
{
PDS_WARNING_IN(_Table)("CDataAccessor::checkAsSetAccessor(): column '%s' points to unallocated object '%s'", _Column->getName().c_str(), index.toString().c_str());
success = false;
continue;
}
if (!checkType(index))
{
PDS_WARNING_IN(_Table)("CDataAccessor::checkAsSetAccessor(): column '%s' points to object '%s' which is not of expected type", _Column->getName().c_str(), index.toString().c_str());
success = false;
continue;
}
}
return success;
}
/*
* Fetch a row into stream
* Fetch the whole object arborescence (i.e. children linked to this object)
* \param row is the row to fetch
* \param data is the stream store data into
*/
bool CTable::fetch(RY_PDS::TRowIndex row, RY_PDS::CPData &data, bool fetchIndex)
{
uint i;
RY_PDS::TTableIndex table = (RY_PDS::TTableIndex)_Id;
RY_PDS::CObjectIndex index(table, row);
if (fetchIndex)
{
data.serial(table, row);
}
CDataAccessor rowaccessor(_Parent, index, 0);
for (i=0; i<_Columns.size(); ++i)
{
if (!_Columns[i].initialised())
{
PDS_WARNING("fetch(): failed, column '%d' not initialised", i);
return false;
}
if (_Columns[i].getMetaType() == PDS_BackRef)
{
// backref are not sent, implicitely deduced
continue;
}
CDataAccessor accessor(rowaccessor, i);
if (!accessor.isValid())
{
PDS_WARNING("fetch(): failed, can't create accessor on column '%d' for object '%s'", i, index.toString().c_str());
return false;
}
if (!accessor.fetch(data))
return false;
}
// lock row
rowaccessor.acquire();
return true;
}
/**
* Fetch column data into stream
*/
bool CTable::CDataAccessor::fetch(RY_PDS::CPData &data)
{
if (!isValid())
{
PDS_WARNING_IN(_Table)("CDataAccessor:fetch(): failed, fetch invalid accessor '%s'", toString().c_str());
return false;
}
switch (_Column->getDataType())
{
case PDS_bool:
case PDS_sint8:
case PDS_uint8:
case PDS_char:
data.serial(*(uint8*)_Data);
break;
case PDS_ucchar:
case PDS_uint16:
case PDS_sint16:
data.serial(*(uint16*)_Data);
break;
case PDS_uint32:
case PDS_sint32:
case PDS_float:
case PDS_CSheetId:
case PDS_CNodeId:
case PDS_enum:
data.serial(*(uint32*)_Data);
break;
case PDS_uint64:
case PDS_sint64:
case PDS_double:
case PDS_CEntityId:
data.serial(*(uint64*)_Data);
break;
case PDS_Index:
{
// check this is a forward ref
// only forward ref are sent, backref are implicitely deduced
if (_Column->getMetaType() == PDS_BackRef)
{
PDS_WARNING_IN(_Table)("CDataAccessor:fetch(): failed, try to serialize '%s' as backref", toString().c_str());
return false;
}
RY_PDS::CObjectIndex child = *(RY_PDS::CObjectIndex*)_Data;
if (!child.isChecksumValid())
{
PDS_WARNING_IN(_Table)("CDataAccessor:fetch(): failed, '%s' points to invalid object '%s'", toString().c_str(), child.toString().c_str());
return false;
}
if (child.isNull())
{
// send object id
RY_PDS::TTableIndex tid = RY_PDS::INVALID_TABLE_INDEX;
RY_PDS::TRowIndex rid = RY_PDS::INVALID_ROW_INDEX;
data.serial(tid, rid);
return true;
}
// send object and hierarchy
return _Table->getParent()->fetch(child, data);
}
break;
case PDS_List:
{
RY_PDS::CSetMap::CAccessor setaccess = getSet();
if (!setaccess.isValid())
{
PDS_WARNING_IN(_Table)("CDataAccessor:fetch(): failed, object '%s' set access is invalid", toString().c_str());
return false;
}
const RY_PDS::TIndexList& list = setaccess.get();
uint i;
for (i=0; i<list.size(); ++i)
{
RY_PDS::CObjectIndex child = list[i];
// fetch object index
RY_PDS::TTableIndex tid = child.table();
RY_PDS::TRowIndex rid = child.row();
data.serial(tid, rid);
// fetch object key, for inset creation
CDataAccessor keyaccess(_Table->getParent(), child, _Attribute->getBackRefKey(), 0);
if (!keyaccess.isValid())
{
PDS_WARNING_IN(_Table)("CDataAccessor:fetch(): failed, '%s' points to invalid object '%s'", toString().c_str(), child.toString().c_str());
return false;
}
keyaccess.fetch(data);
// force table not to fetch index, because we already done it
_Table->getParent()->fetch(child, data, false);
}
RY_PDS::TTableIndex tid = RY_PDS::INVALID_TABLE_INDEX;
RY_PDS::TRowIndex rid = RY_PDS::INVALID_ROW_INDEX;
data.serial(tid, rid);
}
break;
case PDS_dimension:
if (_Column->getByteSize() == 1)
{
data.serial(*(uint8*)_Data);
}
else if (_Column->getByteSize() == 2)
{
data.serial(*(uint16*)_Data);
}
else
{
data.serial(*(uint32*)_Data);
}
break;
}
return true;
}
/*
* Get value as string
*/
string CTable::CDataAccessor::valueAsString(bool expandSet) const
{
if (!isValid())
return string("");
std::string value;
switch (_Column->getDataType())
{
case PDS_bool:
value = dataTypeAsString(*(bool*)_Data);
break;
case PDS_char:
value = dataTypeAsString(*(char*)_Data);
break;
case PDS_uint8:
value = dataTypeAsString(*(uint8*)_Data);
break;
case PDS_ucchar:
case PDS_uint16:
value = dataTypeAsString(*(uint16*)_Data);
break;
case PDS_CSheetId:
case PDS_CNodeId:
case PDS_uint32:
value = dataTypeAsString(*(uint32*)_Data);
break;
case PDS_uint64:
value = dataTypeAsString(*(uint64*)_Data);
break;
case PDS_sint8:
value = dataTypeAsString(*(sint8*)_Data);
break;
case PDS_sint16:
value = dataTypeAsString(*(sint16*)_Data);
break;
case PDS_sint32:
value = dataTypeAsString(*(sint32*)_Data);
break;
case PDS_sint64:
value = dataTypeAsString(*(sint64*)_Data);
break;
case PDS_float:
value = dataTypeAsString(*(float*)_Data);
break;
case PDS_double:
value = dataTypeAsString(*(double*)_Data);
break;
case PDS_CEntityId:
value = dataTypeAsString(*(CEntityId*)_Data);
break;
case PDS_enum:
{
value = "'"+NLMISC::toString(*(uint32*)_Data)+"'";
const CType* type = _Table->getParent()->getType(_Column->getTypeId());
if (type == NULL)
{
value += " <undisplayable type>";
}
else
{
value += " "+type->getName();
if (type->isEnum())
{
value += " "+type->getIndexName(*(uint32*)_Data);
}
else
{
value += " not enum";
}
}
}
break;
case PDS_dimension:
{
if (_Column->getByteSize() == 1)
value = "'"+NLMISC::toString(*(uint8*)_Data)+"'";
else if (_Column->getByteSize() == 2)
value = "'"+NLMISC::toString(*(uint16*)_Data)+"'";
else if (_Column->getByteSize() == 4)
value = "'"+NLMISC::toString(*(uint32*)_Data)+"'";
else
value = "<undisplayable>";
const CType* type = _Table->getParent()->getType(_Column->getTypeId());
if (type == NULL)
{
value += " <undisplayable type>";
}
else
{
value += " "+type->getName();
if (type->isDimension())
{
value += " "+type->getIndexName(*(uint32*)_Data);
}
else
{
value += " not dimension";
}
}
}
break;
case PDS_List:
{
#ifdef DEBUG_DATA_ACCESSOR
if (_IsDebug)
value = "undisplayable set";
else
{
const RY_PDS::TIndexList& list = getSet().get();
if (list.empty())
{
value = "Empty list";
}
else if (expandSet)
{
uint i;
value = "";
for (i=0; i<list.size(); ++i)
value += (list[i].toString()+" ");
}
else
{
value = NLMISC::toString(list.size())+" object(s)";
}
}
#else
const RY_PDS::TIndexList& list = getSet().get();
if (list.empty())
{
value = "Empty list";
}
else
{
value = NLMISC::toString(list.size())+" object(s)";
}
#endif
}
break;
case PDS_Index:
value = "'"+((RY_PDS::CObjectIndex*)_Data)->toString()+"'";
break;
default:
value = "undisplayable value";
break;
}
return value;
}
/*
* Build the index allocator for this table
*/
bool CTable::buildIndexAllocator(RY_PDS::CIndexAllocator& allocat)
{
allocat.clear();
uint row;
for (row=0; row<_TableBuffer.maxRowIndex(); ++row)
{
if (isAllocated(row))
{
allocat.forceAllocated(row);
}
}
return true;
}
/*
* Clear dirty list
*/
bool CTable::clearDirtyList()
{
if (!initialised())
{
PDS_WARNING("clearDirtyList(): table not initialised");
return false;
}
_TableBuffer.resetDirty();
return true;
}
/*
* Reset dirty tags
* Reset all rows so no one is marked as being dirty.
* This method fixes broken list issues
*/
bool CTable::resetDirtyTags()
{
/// \todo fill here
nlstop;
if (!initialised())
{
PDS_WARNING("clearDirtyList(): table not initialised");
return false;
}
return true;
}
/*
* Preload reference files
*/
bool CTable::preloadRefFiles()
{
if (!_TableBuffer.openAllRefFilesRead())
return false;
return true;
}
/*
* Notify new Reference, do necessary job...
*/
bool CTable::notifyNewReference(CRefIndex& newref)
{
if (!initialised())
return false;
_TableBuffer.setupRef(newref);
return true;
}
/*
* Apply delta changes from a file
*/
bool CTable::applyDeltaChanges(const string& filename)
{
if (!initialised())
{
PDS_WARNING("buildDelta(): failed, table not initialised");
return false;
}
return _TableBuffer.applyDeltaChanges(filename);
}
/*
* Reset row to initial value
*/
bool CTable::resetRow(uint8* rowData)
{
// copy row pattern
memcpy(rowData, &(_EmptyRow[0]), _RowSize);
return true;
}
/*
* Build the delta file and purge all dirty rows in this table
*/
bool CTable::buildDelta(const CTimestamp& starttime, const CTimestamp& endtime)
{
H_AUTO(PDS_Table_buildDelta);
if (!initialised())
{
PDS_WARNING("buildDelta(): failed, table not initialised");
return false;
}
return _TableBuffer.buildDelta(starttime, endtime);
}
/*
* Flush table from released rows
*/
bool CTable::flushReleased()
{
if (!initialised())
{
PDS_WARNING("flushReleased(): failed, table not initialised");
return false;
}
_TableBuffer.flushReleased();
return true;
}
/*
* Build RowMapper
*/
bool CTable::buildRowMapper()
{
if (!initialised())
{
PDS_WARNING("buildRowMapper(): failed, table not initialised");
return false;
}
return _TableBuffer.buildRowMapper();
}
/*
* The process row callback, fix forwardrefs from backrefs
*/
bool CTable::processRow(RY_PDS::TTableIndex table, CTableBuffer::CAccessor& accessor)
{
if (table != _Id)
{
PDS_WARNING("processRow(): try to process a row from another table! (id '%d')", table);
return false;
}
RY_PDS::CObjectIndex child = RY_PDS::CObjectIndex(table, accessor.row());
uint i;
for (i=0; i<BackRefInfo.size(); ++i)
{
// get backref accessor
CBackRefFiller& bref = BackRefInfo[i];
CDataAccessor backref(this, accessor, (RY_PDS::TColumnIndex)bref.Column->getId());
RY_PDS::CObjectIndex parent;
if (!backref.isValid() || !backref.getIndex(parent))
{
PDS_WARNING("processRow(): failed to get BackRef index");
return false;
}
// parent not set, do nothing
if (!parent.isValid())
{
continue;
}
// get forwardref accessor
CDataAccessor forwardref(_Parent, parent, bref.Referenced->getId(), 0);
switch (bref.Referenced->getMetaType())
{
case PDS_Set:
processBackRefToSet(forwardref, child);
break;
case PDS_ArrayRef:
// seek to pos in array
if (!forwardref.seek(backref))
{
PDS_WARNING("processRow(): failed to seek into '%s'", forwardref.toString().c_str());
return false;
}
case PDS_ForwardRef:
if (!processBackRefToForwardRef(forwardref, child))
{
PDS_WARNING("processRow(): failed to processBackRefToForwardRef('%s', '%s')", forwardref.toString().c_str(), child.toString().c_str());
return false;
}
break;
default:
break;
}
}
for (i=0; i<ForwardRefInfo.size(); ++i)
{
CAutoForwardRefFiller& fref = ForwardRefInfo[i];
CDataAccessor fwdref(this, accessor, (RY_PDS::TColumnIndex)fref.Column->getId());
RY_PDS::CObjectIndex child;
if (!fwdref.isValid() || !fwdref.getIndex(child))
{
PDS_WARNING("processRow(): failed to get ForwardRef index");
return false;
}
// should not be null and valid
if (child.isValid() && !child.isNull())
continue;
// if potentially broken link
// add to list of rows to fix, and leave
BrokenForwardRefs.push_back(accessor.row());
break;
}
return true;
}
/*
* Process Back Reference to a set
*/
bool CTable::processBackRefToSet(CDataAccessor& parent, RY_PDS::CObjectIndex child)
{
parent.getSet().add(child);
return true;
}
/*
* Process Back Reference to a forward ref
*/
bool CTable::processBackRefToForwardRef(CDataAccessor& parent, RY_PDS::CObjectIndex child)
{
RY_PDS::CObjectIndex checkchild;
if (!parent.getIndex(checkchild))
{
PDS_WARNING("processBackRefToForwardRef(): failed to access to parent '%s'", parent.toString().c_str());
return false;
}
if (checkchild != child)
parent.setIndex(child);
return true;
}
/*
* Fill up Backward and Forward References information
*/
bool CTable::fillRefInfo()
{
uint i;
for (i=0; i<_Columns.size(); ++i)
{
CColumn& column = _Columns[i];
// generate summary of backward reference
if (column.getMetaType() == PDS_BackRef)
{
CTable* parent = _Parent->getNonConstTable((RY_PDS::TTableIndex)(column.getTypeId()));
if (parent == NULL)
return false;
const CAttribute* referenced = parent->getAttribute(column.getParent()->getReferencedAttribute());
if (referenced == NULL)
return false;
CBackRefFiller bref;
bref.Column = &column;
bref.Referenced = referenced;
bref.ParentTable = parent;
BackRefInfo.push_back(bref);
}
// generate summary of forward reference
else if (column.getMetaType() == PDS_ForwardRef)
{
const CAttribute* parent = column.getParent();
// check column in array of ref that are allowed to contain null
if (parent->getMetaType() == PDS_ArrayRef &&
!parent->allowNull())
{
CAutoForwardRefFiller fref;
fref.Column = &column;
ForwardRefInfo.push_back(fref);
}
}
}
return true;
}
/*
* Fix broken forward refs
*/
bool CTable::fixForwardRefs()
{
if (BrokenForwardRefs.empty())
return true;
uint i;
for (i=0; i<BrokenForwardRefs.size(); ++i)
{
RY_PDS::TRowIndex row = BrokenForwardRefs[i];
if (!fixRowForwardRefs(row))
{
PDS_WARNING("fixForwardRefs(): failed to fix forward refs in row '%d' in '%s'", row, _Name.c_str());
}
}
// clean up...
BrokenForwardRefs.clear();
return true;
}
/*
* Fix row broken forward refs
*/
bool CTable::fixRowForwardRefs(RY_PDS::TRowIndex row)
{
// get row and examine all forward refs
CTableBuffer::CAccessor accessor = _TableBuffer.getRow(row);
uint j;
for (j=0; j<ForwardRefInfo.size(); ++j)
{
CAutoForwardRefFiller &colInfo = ForwardRefInfo[j];
CDataAccessor forwardref(this, accessor, (RY_PDS::TColumnIndex)colInfo.Column->getId());
RY_PDS::CObjectIndex index;
if (!forwardref.isValid() || !forwardref.getIndex(index))
{
PDS_WARNING("fixRowForwardRefs(): failed to get forward ref '%s', data left as is", forwardref.toString().c_str());
continue;
}
// was ref fixed?
if (index.isValid())
continue;
// allocate new row
CTable *childTable = _Parent->getNonConstTable((RY_PDS::TTableIndex)(colInfo.Column->getTypeId()));
if (childTable == NULL || !childTable->initialised())
{
PDS_WARNING("fixRowForwardRefs(): failed to get child table '%d'", colInfo.Column->getTypeId());
continue;
}
RY_PDS::TRowIndex alloc = childTable->nextUnallocatedRow();
if (!childTable->allocate(alloc))
{
PDS_WARNING("fixRowForwardRefs(): failed to get allocate free row in table '%s'", childTable->getName().c_str());
continue;
}
// get an accessor on row key
CTableBuffer::CAccessor allocAccess = childTable->_TableBuffer.getRow(alloc);
CDataAccessor keyAccess(childTable, allocAccess, (RY_PDS::TColumnIndex)(childTable->getAttribute(childTable->getKey())->getOffset()));
if (!keyAccess.isValid())
{
PDS_WARNING("fixRowForwardRefs(): failed to get key access of row '%s:%d'", childTable->getName().c_str(), alloc);
childTable->_TableBuffer.releaseRow(allocAccess);
continue;
}
// compute key
TEnumValue keyValue = forwardref.column()->getId()-forwardref.attribute()->getOffset();
// set new row key
if (!keyAccess.setAsIndexType(keyValue))
{
PDS_WARNING("fixRowForwardRefs(): failed to set key '%s'", keyAccess.toString().c_str());
childTable->_TableBuffer.releaseRow(allocAccess);
continue;
}
CDataAccessor backref(keyAccess, (RY_PDS::TColumnIndex)(childTable->getAttribute(forwardref.attribute()->getReferencedAttribute())->getOffset()));
// set back&forward links
forwardref.setIndex(backref.getObjectIndex());
backref.setIndex(forwardref.getObjectIndex());
// fixup links in new child
if (!childTable->fixRowForwardRefs(alloc))
{
PDS_WARNING("fixRowForwardRefs(): failed to set fix forward refs in newly allocated row '%s'", backref.getObjectIndex().toString().c_str());
childTable->_TableBuffer.releaseRow(allocAccess);
continue;
}
childTable->_TableBuffer.releaseRow(allocAccess);
}
_TableBuffer.releaseRow(accessor);
return true;
}
/*
* Dump accessor content and info to xml
*/
void CTable::CDataAccessor::dumpToXml(NLMISC::IStream& xml, sint expandDepth)
{
if (xml.isReading())
return;
xml.xmlPushBegin("value");
xml.xmlSetAttrib("valid");
bool valid = isValid();
xml.serial(valid);
bool closeTag = true;
if (valid)
{
xml.xmlSetAttrib("name");
std::string columnName = _Column->getName();
xml.serial(columnName);
xml.xmlSetAttrib("type");
std::string typeName = getNameFromDataType(_Column->getDataType());
xml.serial(typeName);
std::string value;
xml.xmlSetAttrib("value");
switch (_Column->getDataType())
{
case PDS_bool:
xml.serial(*(bool*)_Data);
break;
case PDS_char:
xml.serial(*(char*)_Data);
break;
case PDS_uint8:
xml.serial(*(uint8*)_Data);
break;
case PDS_ucchar:
case PDS_uint16:
xml.serial(*(uint16*)_Data);
break;
case PDS_CSheetId:
case PDS_CNodeId:
case PDS_uint32:
xml.serial(*(uint32*)_Data);
break;
case PDS_uint64:
xml.serial(*(uint64*)_Data);
break;
case PDS_sint8:
xml.serial(*(sint8*)_Data);
break;
case PDS_sint16:
xml.serial(*(sint16*)_Data);
break;
case PDS_sint32:
xml.serial(*(sint32*)_Data);
break;
case PDS_sint64:
xml.serial(*(sint64*)_Data);
break;
case PDS_float:
xml.serial(*(float*)_Data);
break;
case PDS_double:
xml.serial(*(double*)_Data);
break;
case PDS_CEntityId:
{
std::string id = ((CEntityId*)_Data)->toString();
xml.serial(id);
}
break;
case PDS_enum:
{
std::string value;
const CType* type = _Table->getParent()->getType(_Column->getTypeId());
if (type == NULL || !type->isEnum())
{
xml.serial(*(uint32*)_Data);
}
else
{
std::string name = type->getIndexName(*(uint32*)_Data);
xml.serial(name);
}
}
break;
case PDS_dimension:
{
if (_Column->getByteSize() == 1)
{
xml.serial(*(uint8*)_Data);
}
else if (_Column->getByteSize() == 2)
{
xml.serial(*(uint16*)_Data);
}
else if (_Column->getByteSize() == 4)
{
xml.serial(*(uint32*)_Data);
}
else
{
std::string unknown = "non displayable";
xml.serial(unknown);
}
}
break;
case PDS_List:
{
std::string value = "list";
xml.serial(value);
xml.xmlPushEnd();
closeTag = false;
if (expandDepth != 0)
{
const RY_PDS::TIndexList& list = getSet().get();
if (!list.empty())
{
uint i;
for (i=0; i<list.size(); ++i)
_Table->_Parent->dumpToXml(list[i], xml, expandDepth-1);
}
}
}
break;
case PDS_Index:
{
std::string value = ((RY_PDS::CObjectIndex*)_Data)->toString(_Table->getParent());
xml.serial(value);
xml.xmlPushEnd();
closeTag = false;
if (expandDepth != 0 && _Attribute->getMetaType() != PDS_BackRef)
_Table->_Parent->dumpToXml(*(RY_PDS::CObjectIndex*)_Data, xml, expandDepth-1);
}
break;
default:
{
std::string unknown = "non displayable";
xml.serial(unknown);
}
break;
}
}
if (closeTag)
xml.xmlPushEnd();
xml.xmlPop();
}
/*
* Dump accessor content and info to xml
*/
void CTable::dumpToXml(RY_PDS::TRowIndex row, NLMISC::IStream& xml, sint expandDepth)
{
if (xml.isReading())
return;
xml.xmlPushBegin("object");
xml.xmlSetAttrib("valid");
bool valid = initialised();
xml.serial(valid);
if (initialised())
{
xml.xmlSetAttrib("name");
xml.serial(_Name);
xml.xmlSetAttrib("index");
RY_PDS::CObjectIndex index((RY_PDS::TTableIndex)_Id, (RY_PDS::TRowIndex)row);
std::string indexName = index.toString(_Parent);
xml.serial(indexName);
CTableBuffer::CAccessor rowaccess = _TableBuffer.getRow(row);
bool rowAllocated = rowaccess.allocated();
xml.xmlSetAttrib("allocated");
xml.serial(rowAllocated);
xml.xmlPushEnd();
if (rowAllocated)
{
uint i;
for (i=0; i<_Columns.size(); ++i)
{
CDataAccessor accessor(this, rowaccess, i);
const CColumn &col = _Columns[i];
accessor.dumpToXml(xml, expandDepth);
}
}
_TableBuffer.releaseRow(rowaccess);
}
else
{
xml.xmlPushEnd();
}
xml.xmlPop();
}